├── .eslintignore ├── .eslintrc.yml ├── .gitignore ├── README.md ├── docker-compose.yml ├── docs ├── azure.md ├── docker-compose.md └── minikube.md ├── helm ├── Chart.yaml └── charts │ ├── api-gateway │ ├── Chart.yaml │ ├── templates │ │ ├── deployment-redis.yaml │ │ ├── deployment.yaml │ │ ├── service-redis.yaml │ │ └── service.yaml │ └── values.yaml │ ├── jaeger │ ├── Chart.yaml │ ├── templates │ │ ├── deployment.yaml │ │ ├── service-agent.yaml │ │ ├── service-collector.yaml │ │ ├── service-query.yaml │ │ └── service-zipkin.yaml │ └── values.yaml │ ├── user-api │ ├── Chart.yaml │ ├── templates │ │ ├── deployment.yaml │ │ └── service.yaml │ └── values.yaml │ └── vehicle-api │ ├── Chart.yaml │ ├── templates │ ├── deployment.yaml │ └── service.yaml │ └── values.yaml ├── images ├── swagger_api_gateway.png ├── tracing_output.png └── webinar-microservices-diagram.png ├── package-lock.json ├── package.json ├── scripts └── docker-push.sh └── services ├── api-gateway ├── .dockerignore ├── Dockerfile ├── api │ ├── controllers │ │ ├── healthzController.js │ │ └── userController.js │ └── swagger │ │ └── swagger.yaml ├── app.js ├── config │ └── default.yaml ├── index.js ├── package-lock.json ├── package.json └── server.js ├── user-api ├── .dockerignore ├── Dockerfile ├── api │ ├── controllers │ │ ├── healthzController.js │ │ └── userController.js │ └── swagger │ │ └── swagger.yaml ├── app.js ├── config │ └── default.yaml ├── index.js ├── package-lock.json ├── package.json └── server.js └── vehicle-api ├── .dockerignore ├── Dockerfile ├── api ├── controllers │ ├── healthzController.js │ └── vehicleController.js └── swagger │ └── swagger.yaml ├── app.js ├── config └── default.yaml ├── index.js ├── package-lock.json ├── package.json └── server.js /.eslintignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | build 3 | -------------------------------------------------------------------------------- /.eslintrc.yml: -------------------------------------------------------------------------------- 1 | --- 2 | extends: airbnb 3 | env: 4 | node: true 5 | mocha: true 6 | es6: true 7 | parserOptions: 8 | sourceType: module 9 | rules: 10 | no-console: 0 11 | generator-star-spacing: 12 | - 2 13 | - before: true 14 | after: true 15 | no-shadow: 0 16 | require-yield: 0 17 | no-param-reassign: 0 18 | comma-dangle: 19 | - error 20 | - never 21 | no-underscore-dangle: 0 22 | import/no-extraneous-dependencies: 23 | - 2 24 | - devDependencies: true 25 | import/order: 26 | - error 27 | func-names: 0 28 | no-unused-expressions: 0 29 | prefer-arrow-callback: 1 30 | no-use-before-define: 31 | - 2 32 | - functions: false 33 | space-before-function-paren: 34 | - 2 35 | - always 36 | max-len: 37 | - 2 38 | - 120 39 | - 2 40 | semi: 41 | - 2 42 | - never 43 | strict: 0 44 | arrow-parens: 45 | - 2 46 | - always 47 | jsx-a11y/href-no-hash: 0 48 | react/jsx-filename-extension: 0 49 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # dependencies 2 | node_modules 3 | 4 | # testing 5 | coverage 6 | .nyc_output 7 | 8 | # production 9 | build 10 | 11 | *.log 12 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Microservices​ ​with​ ​Node.js​ ​and​ ​Kubernetes​ 2 | 3 | Demonstrating a HTTP based microservices and API gateway with Kubernetes and Helm. 4 | 5 | ## Architecture 6 | 7 | ![Architecture](images/webinar-microservices-diagram.png) 8 | 9 | ## Getting Started 10 | 11 | - [Azure Container Service](/docs/azure.md) 12 | - [Docker Compose](/docs/docker-compose.md) 13 | - [Minikube](/docs/minikube.md) 14 | 15 | ## Used technologies 16 | 17 | - [Node.js](https://nodejs.org/en/) 18 | - [Swagger](https://swagger.io) 19 | - [Kubernetes](https://kubernetes.io/) 20 | - [Helm](https://github.com/kubernetes/helm) 21 | 22 | ## Docs 23 | 24 | http://localhost:3000/docs 25 | 26 | ![Swagger API Gateway](images/swagger_api_gateway.png) 27 | 28 | ## Distributed Tracing 29 | 30 | http://localhost:16686 31 | 32 | ![Distributed Tracing](images/tracing_output.png) 33 | -------------------------------------------------------------------------------- /docker-compose.yml: -------------------------------------------------------------------------------- 1 | version: '3.3' 2 | services: 3 | redis: 4 | image: redis:4.0.1-alpine 5 | jaeger: 6 | ports: 7 | - "5775:5775/udp" 8 | - "6831:6831/udp" 9 | - "6832:6832/udp" # default UDP reporter port 10 | - "16686:16686" # UI port 11 | - "14268:14268" 12 | image: jaegertracing/all-in-one:latest 13 | api-gateway: 14 | depends_on: 15 | - redis 16 | - user-api 17 | - vehicle-api 18 | build: services/api-gateway 19 | restart: on-failure 20 | ports: 21 | - "3000:3000" #entrypoint 22 | environment: 23 | JAEGER_HOST: jaeger 24 | REDIS_URI: redis://redis:6379 25 | USER_API_SERVICE_HOST: user-api 26 | USER_API_SERVICE_PORT: 3000 27 | VEHICLE_API_SERVICE_HOST: vehicle-api 28 | VEHICLE_API_SERVICE_PORT: 3000 29 | user-api: 30 | build: services/user-api 31 | restart: on-failure 32 | ports: 33 | - "3001:3000" # exposed for debugging 34 | environment: 35 | JAEGER_HOST: jaeger 36 | vehicle-api: 37 | build: services/vehicle-api 38 | restart: on-failure 39 | ports: 40 | - "3002:3000" # exposed for debugging 41 | environment: 42 | JAEGER_HOST: jaeger 43 | -------------------------------------------------------------------------------- /docs/azure.md: -------------------------------------------------------------------------------- 1 | # Getting Started With Azure Container Service 2 | 3 | ## 1. Creating Kubernetes Cluster 4 | 5 | ```sh 6 | # Create a resource group 7 | az group create --name webinar-microservices --location westeurope 8 | 9 | # Provision a new Kubernetes cluster 10 | az acs create --resource-group webinar-microservices --name my-services --generate-ssh-keys --orchestrator-type kubernetes 11 | 12 | # Configure kubectl with the new cluster 13 | az acs kubernetes get-credentials --resource-group webinar-microservices --name my-services --ssh-key-file ~/.ssh/id_rsa 14 | ``` 15 | 16 | For more information check out the official [Azure Container Service with Kubernetes Documentation](https://docs.microsoft.com/en-us/azure/container-service/kubernetes/). 17 | 18 | ## 2. Installing Helm chart 19 | 20 | Azure Container Services comes with a pre-installed helm tiller. 21 | You should be sure that your local helm matches the version of the server: `helm version` 22 | 23 | ```sh 24 | helm init 25 | helm install --name services ./helm 26 | ``` 27 | 28 | You can expose the `api-gateway` service via `LoadBalancer`: 29 | 30 | ```sh 31 | helm upgrade services ./helm/ --set api-gateway.public=true 32 | ``` 33 | 34 | To read more about helm check out the [Packing a Kubernetes Microservices App with Helm on Azure Container Service](https://open.microsoft.com/2017/05/23/kubernetes-helm-microsoft-azure-container-service/) article. 35 | 36 | ## 3. Accessing Application 37 | 38 | ```sh 39 | kubectl port-forward $(kubectl get pods | grep api-gateway | head -1 | cut -f1 -d' ') 3000:3000 40 | ``` 41 | 42 | http://localhost:3000 43 | 44 | ## 4. Accessing Jaeger 45 | 46 | ```sh 47 | kubectl port-forward $(kubectl get pods | grep jaeger-deployment | head -1 | cut -f1 -d' ') 16686:16686 48 | ``` 49 | 50 | http://localhost:16686 51 | -------------------------------------------------------------------------------- /docs/docker-compose.md: -------------------------------------------------------------------------------- 1 | # Getting Started With Docker Compose 2 | 3 | ## 1. Starting Services 4 | 5 | ```sh 6 | docker-compose build 7 | docker-compose up 8 | ``` 9 | 10 | ## 2. Accessing Application 11 | 12 | http://localhost:3000/docs 13 | 14 | ## 3. Accessing Jaeger 15 | 16 | http://localhost:16686 17 | -------------------------------------------------------------------------------- /docs/minikube.md: -------------------------------------------------------------------------------- 1 | # Getting Started With Minikube 2 | 3 | ## 1. Building Docker image 4 | 5 | Installing [Minikube](https://github.com/kubernetes/minikube) and [Helm](https://github.com/kubernetes/helm) 6 | 7 | ```sh 8 | brew install kubernetes-helm 9 | brew cask install minikube 10 | ``` 11 | 12 | Re-using local Docker daemon with minikube: 13 | 14 | ```sh 15 | eval $(minikube docker-env) 16 | ``` 17 | 18 | Building Docker image: 19 | 20 | ```sh 21 | ./scripts/docker-build.sh 22 | ``` 23 | 24 | ## 2. Installing Helm chart 25 | 26 | ```sh 27 | helm init 28 | helm install --name services ./helm 29 | ``` 30 | 31 | ## 3. Accessing Application 32 | 33 | ```sh 34 | minikube service api-gateway 35 | minikube service jaeger-query 36 | ``` 37 | -------------------------------------------------------------------------------- /helm/Chart.yaml: -------------------------------------------------------------------------------- 1 | apiVersion: v1 2 | description: Microservices with Node.js and Kubernetes 3 | name: services 4 | version: 0.1.0 5 | -------------------------------------------------------------------------------- /helm/charts/api-gateway/Chart.yaml: -------------------------------------------------------------------------------- 1 | apiVersion: v1 2 | description: api-gateway service 3 | name: api-gateway 4 | version: 0.1.0 5 | -------------------------------------------------------------------------------- /helm/charts/api-gateway/templates/deployment-redis.yaml: -------------------------------------------------------------------------------- 1 | apiVersion: extensions/v1beta1 2 | kind: Deployment 3 | metadata: 4 | name: {{ .Chart.Name }}-redis 5 | labels: 6 | chart: {{ .Chart.Name }}-{{ .Chart.Version }} 7 | heritage: {{ .Release.Service }} 8 | release: {{ .Release.Name }} 9 | spec: 10 | template: 11 | metadata: 12 | labels: 13 | app: {{ .Chart.Name }}-redis 14 | spec: 15 | containers: 16 | - command: 17 | - redis-server 18 | - --appendonly 19 | - "yes" 20 | image: redis:4.0.1-alpine 21 | name: {{ .Chart.Name }}-redis 22 | ports: 23 | - containerPort: 6379 24 | name: redis 25 | protocol: TCP 26 | resources: 27 | limits: 28 | cpu: 120m 29 | memory: 386Mi 30 | requests: 31 | cpu: 100m 32 | memory: 256Mi 33 | livenessProbe: 34 | exec: 35 | command: 36 | - redis-cli 37 | - ping 38 | initialDelaySeconds: 30 39 | timeoutSeconds: 5 40 | readinessProbe: 41 | exec: 42 | command: 43 | - redis-cli 44 | - ping 45 | initialDelaySeconds: 5 46 | timeoutSeconds: 1 47 | -------------------------------------------------------------------------------- /helm/charts/api-gateway/templates/deployment.yaml: -------------------------------------------------------------------------------- 1 | apiVersion: apps/v1beta1 2 | kind: Deployment 3 | metadata: 4 | name: {{ .Chart.Name }} 5 | labels: 6 | chart: {{ .Chart.Name }}-{{ .Chart.Version }} 7 | heritage: {{ .Release.Service }} 8 | release: {{ .Release.Name }} 9 | spec: 10 | replicas: 2 11 | template: 12 | metadata: 13 | labels: 14 | app: {{ .Chart.Name }} 15 | spec: 16 | containers: 17 | - name: {{ .Chart.Name }} 18 | image: {{ .Values.image }} 19 | env: 20 | - name: JAEGER_HOST 21 | value: jaeger-agent.{{ .Release.Namespace }} 22 | - name: REDIS_URI 23 | value: redis://{{ .Chart.Name }}-redis.{{ .Release.Namespace }}:6379 24 | ports: 25 | - containerPort: 3000 26 | name: server 27 | resources: 28 | requests: 29 | memory: 100Mi 30 | cpu: 50m 31 | limits: 32 | memory: 1536Mi 33 | cpu: 55m 34 | -------------------------------------------------------------------------------- /helm/charts/api-gateway/templates/service-redis.yaml: -------------------------------------------------------------------------------- 1 | apiVersion: v1 2 | kind: Service 3 | metadata: 4 | labels: 5 | chart: {{ .Chart.Name }}-{{ .Chart.Version }} 6 | heritage: {{ .Release.Service }} 7 | release: {{ .Release.Name }} 8 | name: {{ .Chart.Name }}-redis 9 | spec: 10 | ports: 11 | - port: 6379 12 | protocol: TCP 13 | targetPort: redis 14 | selector: 15 | app: {{ .Chart.Name }}-redis 16 | type: ClusterIP 17 | -------------------------------------------------------------------------------- /helm/charts/api-gateway/templates/service.yaml: -------------------------------------------------------------------------------- 1 | apiVersion: v1 2 | kind: Service 3 | metadata: 4 | name: {{ .Chart.Name }} 5 | labels: 6 | chart: {{ .Chart.Name }}-{{ .Chart.Version }} 7 | heritage: {{ .Release.Service }} 8 | release: {{ .Release.Name }} 9 | spec: 10 | ports: 11 | - port: 80 12 | targetPort: server 13 | protocol: TCP 14 | selector: 15 | app: {{ .Chart.Name }} 16 | type: {{ if .Values.public }}LoadBalancer{{ else }}NodePort{{ end }} 17 | -------------------------------------------------------------------------------- /helm/charts/api-gateway/values.yaml: -------------------------------------------------------------------------------- 1 | image: registry.hub.docker.com/risingstack/webinar-ms-api-gateway:v1 2 | public: false 3 | -------------------------------------------------------------------------------- /helm/charts/jaeger/Chart.yaml: -------------------------------------------------------------------------------- 1 | apiVersion: v1 2 | description: Jaeger distributed tracing 3 | name: jaeger 4 | version: 0.1.0 5 | -------------------------------------------------------------------------------- /helm/charts/jaeger/templates/deployment.yaml: -------------------------------------------------------------------------------- 1 | apiVersion: extensions/v1beta1 2 | kind: Deployment 3 | metadata: 4 | name: jaeger-deployment 5 | labels: 6 | app: {{ .Chart.Name }} 7 | jaeger-infra: {{ .Chart.Name }}-deployment 8 | chart: {{ .Chart.Name }}-{{ .Chart.Version }} 9 | heritage: {{ .Release.Service }} 10 | release: {{ .Release.Name }} 11 | spec: 12 | replicas: 1 13 | strategy: 14 | type: Recreate 15 | template: 16 | metadata: 17 | labels: 18 | app: {{ .Chart.Name }} 19 | jaeger-infra: {{ .Chart.Name }}-pod 20 | spec: 21 | containers: 22 | - env: 23 | - name: COLLECTOR_ZIPKIN_HTTP_PORT 24 | value: "9411" 25 | image: "{{ .Values.image.repository }}" 26 | name: {{ .Chart.Name }} 27 | ports: 28 | - containerPort: 5775 29 | protocol: UDP 30 | - containerPort: 6831 31 | protocol: UDP 32 | - containerPort: 6832 33 | protocol: UDP 34 | - containerPort: 16686 35 | protocol: TCP 36 | - containerPort: 9411 37 | protocol: TCP 38 | readinessProbe: 39 | httpGet: 40 | path: "/" 41 | port: 16686 42 | initialDelaySeconds: 5 43 | -------------------------------------------------------------------------------- /helm/charts/jaeger/templates/service-agent.yaml: -------------------------------------------------------------------------------- 1 | apiVersion: v1 2 | kind: Service 3 | metadata: 4 | name: {{ .Chart.Name }}-agent 5 | labels: 6 | app: {{ .Chart.Name }} 7 | jaeger-infra: agent-service 8 | chart: {{ .Chart.Name }}-{{ .Chart.Version }} 9 | heritage: {{ .Release.Service }} 10 | release: {{ .Release.Name }} 11 | spec: 12 | ports: 13 | - name: agent-zipkin-thrift 14 | port: 5775 15 | protocol: UDP 16 | targetPort: 5775 17 | - name: agent-compact 18 | port: 6831 19 | protocol: UDP 20 | targetPort: 6831 21 | - name: agent-binary 22 | port: 6832 23 | protocol: UDP 24 | targetPort: 6832 25 | clusterIP: None 26 | selector: 27 | jaeger-infra: {{ .Chart.Name }}-pod 28 | -------------------------------------------------------------------------------- /helm/charts/jaeger/templates/service-collector.yaml: -------------------------------------------------------------------------------- 1 | apiVersion: v1 2 | kind: Service 3 | metadata: 4 | name: {{ .Chart.Name }}-collector 5 | labels: 6 | app: {{ .Chart.Name }} 7 | jaeger-infra: collector-service 8 | chart: {{ .Chart.Name }}-{{ .Chart.Version }} 9 | heritage: {{ .Release.Service }} 10 | release: {{ .Release.Name }} 11 | spec: 12 | ports: 13 | - name: {{ .Chart.Name }}-collector-tchannel 14 | port: 14267 15 | protocol: TCP 16 | targetPort: 14267 17 | - name: {{ .Chart.Name }}-collector-http 18 | port: 14268 19 | protocol: TCP 20 | targetPort: 14268 21 | - name: {{ .Chart.Name }}-collector-zipkin 22 | port: 9411 23 | protocol: TCP 24 | targetPort: 9411 25 | selector: 26 | jaeger-infra: {{ .Chart.Name }}-pod 27 | type: ClusterIP 28 | -------------------------------------------------------------------------------- /helm/charts/jaeger/templates/service-query.yaml: -------------------------------------------------------------------------------- 1 | apiVersion: v1 2 | kind: Service 3 | metadata: 4 | name: {{ .Chart.Name }}-query 5 | labels: 6 | app: {{ .Chart.Name }} 7 | jaeger-infra: {{ .Chart.Name }}-service 8 | chart: {{ .Chart.Name }}-{{ .Chart.Version }} 9 | heritage: {{ .Release.Service }} 10 | release: {{ .Release.Name }} 11 | spec: 12 | ports: 13 | - name: query-http 14 | port: 80 15 | protocol: TCP 16 | targetPort: 16686 17 | selector: 18 | jaeger-infra: {{ .Chart.Name }}-pod 19 | type: NodePort 20 | -------------------------------------------------------------------------------- /helm/charts/jaeger/templates/service-zipkin.yaml: -------------------------------------------------------------------------------- 1 | apiVersion: v1 2 | kind: Service 3 | metadata: 4 | name: zipkin 5 | labels: 6 | app: {{ .Chart.Name }} 7 | jaeger-infra: zipkin-service 8 | chart: {{ .Chart.Name }}-{{ .Chart.Version }} 9 | heritage: {{ .Release.Service }} 10 | release: {{ .Release.Name }} 11 | spec: 12 | ports: 13 | - name: {{ .Chart.Name }}-collector-zipkin 14 | port: 9411 15 | protocol: TCP 16 | targetPort: 9411 17 | clusterIP: None 18 | selector: 19 | jaeger-infra: {{ .Chart.Name }}-pod 20 | -------------------------------------------------------------------------------- /helm/charts/jaeger/values.yaml: -------------------------------------------------------------------------------- 1 | image: 2 | repository: jaegertracing/all-in-one 3 | -------------------------------------------------------------------------------- /helm/charts/user-api/Chart.yaml: -------------------------------------------------------------------------------- 1 | apiVersion: v1 2 | description: user-api service 3 | name: user-api 4 | version: 0.1.0 5 | -------------------------------------------------------------------------------- /helm/charts/user-api/templates/deployment.yaml: -------------------------------------------------------------------------------- 1 | apiVersion: apps/v1beta1 2 | kind: Deployment 3 | metadata: 4 | name: {{ .Chart.Name }} 5 | labels: 6 | chart: {{ .Chart.Name }}-{{ .Chart.Version }} 7 | heritage: {{ .Release.Service }} 8 | release: {{ .Release.Name }} 9 | spec: 10 | replicas: 2 11 | template: 12 | metadata: 13 | labels: 14 | app: {{ .Chart.Name }} 15 | spec: 16 | containers: 17 | - name: {{ .Chart.Name }} 18 | image: {{ .Values.image }} 19 | env: 20 | - name: JAEGER_HOST 21 | value: jaeger-agent.{{ .Release.Namespace }} 22 | ports: 23 | - containerPort: 3000 24 | name: server 25 | resources: 26 | requests: 27 | memory: 100Mi 28 | cpu: 50m 29 | limits: 30 | memory: 1536Mi 31 | cpu: 55m 32 | -------------------------------------------------------------------------------- /helm/charts/user-api/templates/service.yaml: -------------------------------------------------------------------------------- 1 | apiVersion: v1 2 | kind: Service 3 | metadata: 4 | name: {{ .Chart.Name }} 5 | labels: 6 | chart: {{ .Chart.Name }}-{{ .Chart.Version }} 7 | heritage: {{ .Release.Service }} 8 | release: {{ .Release.Name }} 9 | spec: 10 | ports: 11 | - port: 80 12 | targetPort: server 13 | protocol: TCP 14 | selector: 15 | app: {{ .Chart.Name }} 16 | type: ClusterIP 17 | -------------------------------------------------------------------------------- /helm/charts/user-api/values.yaml: -------------------------------------------------------------------------------- 1 | image: registry.hub.docker.com/risingstack/webinar-ms-user-api:v1 2 | -------------------------------------------------------------------------------- /helm/charts/vehicle-api/Chart.yaml: -------------------------------------------------------------------------------- 1 | apiVersion: v1 2 | description: vehicle-api service 3 | name: vehicle-api 4 | version: 0.1.0 5 | -------------------------------------------------------------------------------- /helm/charts/vehicle-api/templates/deployment.yaml: -------------------------------------------------------------------------------- 1 | apiVersion: apps/v1beta1 2 | kind: Deployment 3 | metadata: 4 | name: {{ .Chart.Name }} 5 | labels: 6 | chart: {{ .Chart.Name }}-{{ .Chart.Version }} 7 | heritage: {{ .Release.Service }} 8 | release: {{ .Release.Name }} 9 | spec: 10 | replicas: 2 11 | template: 12 | metadata: 13 | labels: 14 | app: {{ .Chart.Name }} 15 | spec: 16 | containers: 17 | - name: {{ .Chart.Name }} 18 | image: {{ .Values.image }} 19 | env: 20 | - name: JAEGER_HOST 21 | value: jaeger-agent.{{ .Release.Namespace }} 22 | ports: 23 | - containerPort: 3000 24 | name: server 25 | resources: 26 | requests: 27 | memory: 100Mi 28 | cpu: 50m 29 | limits: 30 | memory: 1536Mi 31 | cpu: 55m 32 | -------------------------------------------------------------------------------- /helm/charts/vehicle-api/templates/service.yaml: -------------------------------------------------------------------------------- 1 | apiVersion: v1 2 | kind: Service 3 | metadata: 4 | name: {{ .Chart.Name }} 5 | labels: 6 | chart: {{ .Chart.Name }}-{{ .Chart.Version }} 7 | heritage: {{ .Release.Service }} 8 | release: {{ .Release.Name }} 9 | spec: 10 | ports: 11 | - port: 80 12 | targetPort: server 13 | protocol: TCP 14 | selector: 15 | app: {{ .Chart.Name }} 16 | type: ClusterIP 17 | -------------------------------------------------------------------------------- /helm/charts/vehicle-api/values.yaml: -------------------------------------------------------------------------------- 1 | image: registry.hub.docker.com/risingstack/webinar-ms-vehicle-api:v1 2 | -------------------------------------------------------------------------------- /images/swagger_api_gateway.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/RisingStack/webinar-kubernetes-api-gateway/6383d52f7a088eb83e5fed44a65421d99e2f5780/images/swagger_api_gateway.png -------------------------------------------------------------------------------- /images/tracing_output.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/RisingStack/webinar-kubernetes-api-gateway/6383d52f7a088eb83e5fed44a65421d99e2f5780/images/tracing_output.png -------------------------------------------------------------------------------- /images/webinar-microservices-diagram.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/RisingStack/webinar-kubernetes-api-gateway/6383d52f7a088eb83e5fed44a65421d99e2f5780/images/webinar-microservices-diagram.png -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "webinar-microservices", 3 | "version": "0.0.1", 4 | "private": true, 5 | "description": "Microservices​ ​with​ ​Node.js​ ​and​ ​Kubernetes​", 6 | "scripts": { 7 | "lint": "eslint src" 8 | }, 9 | "devDependencies": { 10 | "eslint": "4.4.1", 11 | "eslint-config-airbnb": "15.1.0", 12 | "eslint-config-airbnb-base": "11.3.1", 13 | "eslint-plugin-import": "2.7.0", 14 | "eslint-plugin-jsx-a11y": "6.0.2", 15 | "eslint-plugin-promise": "3.5.0", 16 | "eslint-plugin-react": "7.2.0" 17 | }, 18 | "dependencies": {} 19 | } 20 | -------------------------------------------------------------------------------- /scripts/docker-push.sh: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env bash 2 | 3 | # Builds and pushes Docker images to DockerHub 4 | 5 | TAG=$1 6 | TAG=${TAG:-"v1"} 7 | 8 | cd ./services/api-gateway 9 | docker build -t risingstack/webinar-ms-api-gateway:latest -t risingstack/webinar-ms-api-gateway:$TAG . 10 | docker push risingstack/webinar-ms-api-gateway 11 | 12 | cd ../user-api 13 | docker build -t risingstack/webinar-ms-user-api:latest -t risingstack/webinar-ms-user-api:$TAG . 14 | docker push risingstack/webinar-ms-user-api 15 | 16 | cd ../vehicle-api 17 | docker build -t risingstack/webinar-ms-vehicle-api:latest -t risingstack/webinar-ms-vehicle-api:$TAG . 18 | docker push risingstack/webinar-ms-vehicle-api 19 | -------------------------------------------------------------------------------- /services/api-gateway/.dockerignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | -------------------------------------------------------------------------------- /services/api-gateway/Dockerfile: -------------------------------------------------------------------------------- 1 | FROM risingstack/alpine:3.4-v8.5.0-4.7.0 2 | 3 | ENV PORT 3000 4 | 5 | EXPOSE 3000 6 | 7 | COPY package.json package.json 8 | RUN npm install 9 | 10 | COPY . . 11 | 12 | CMD ["node", "."] 13 | -------------------------------------------------------------------------------- /services/api-gateway/api/controllers/healthzController.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | /** 4 | * List users via user-api and vehicle-api 5 | * @function get 6 | * @param {express.Request} req 7 | * @param {express.Response} res 8 | * @param {Function} next 9 | */ 10 | function get (req, res, next) { 11 | res.json({ 12 | status: 'ok' 13 | }) 14 | next() 15 | } 16 | 17 | module.exports = { 18 | get 19 | } 20 | -------------------------------------------------------------------------------- /services/api-gateway/api/controllers/userController.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | const assert = require('assert') 4 | const logger = require('winston') 5 | const SwaggerClient = require('swagger-client') 6 | 7 | assert(process.env.USER_API_SERVICE_HOST, 'USER_API_SERVICE_HOST is required') 8 | assert(process.env.USER_API_SERVICE_PORT, 'USER_API_SERVICE_PORT is required') 9 | assert(process.env.VEHICLE_API_SERVICE_HOST, 'VEHICLE_API_SERVICE_HOST is required') 10 | assert(process.env.VEHICLE_API_SERVICE_PORT, 'VEHICLE_API_SERVICE_PORT is required') 11 | 12 | const userAPIUri = `http://${process.env.USER_API_SERVICE_HOST}:${process.env.USER_API_SERVICE_PORT}` 13 | const vehicleAPIUri = `http://${process.env.VEHICLE_API_SERVICE_HOST}:${process.env.VEHICLE_API_SERVICE_PORT}` 14 | 15 | let userClient 16 | let vehicleClient 17 | 18 | /** 19 | * List users via user-api and vehicle-api 20 | * @function get 21 | * @param {express.Request} req 22 | * @param {express.Response} res 23 | * @param {Function} next 24 | */ 25 | async function get (req, res, next) { 26 | const User = await getUserClient() 27 | const { obj: users } = await User.apis.user.get() 28 | logger.debug('Get users', users) 29 | 30 | // Fetch vehicles by user ids 31 | const userIds = users.map((user) => user.id) 32 | const Vehicle = await getVehicleClient() 33 | const { obj: vehicles } = await Vehicle.apis.vehicle.get({ userIds }) 34 | 35 | // Map vehicles by user ids 36 | const vehiclesByUserId = vehicles.reduce((vehiclesByUserId, vehicle) => { 37 | vehiclesByUserId[vehicle.userId] = vehiclesByUserId[vehicle.userId] || [] 38 | vehiclesByUserId[vehicle.userId].push(vehicle) 39 | return vehiclesByUserId 40 | }, {}) 41 | logger.debug('Get vehicles', vehicles) 42 | 43 | // Response with users and vehicles together 44 | const responseUsers = users.map((user) => { 45 | const userVehicles = vehiclesByUserId[user.id] || [] 46 | 47 | return Object.assign(user, { 48 | vehicles: userVehicles.map((vehicle) => ({ 49 | id: vehicle.id, 50 | name: vehicle.name 51 | })) 52 | }) 53 | }) 54 | 55 | res.json(responseUsers) 56 | next() 57 | } 58 | 59 | /** 60 | * Singleton that resolves User API client 61 | * @function getUserClient 62 | * @param {Promise} client - swagger client 63 | */ 64 | async function getUserClient () { 65 | if (!userClient) { 66 | logger.debug('User Client resolve') 67 | userClient = await SwaggerClient(`${userAPIUri}/api-docs`) 68 | logger.debug('User Client succesfully resolved') 69 | } 70 | return userClient 71 | } 72 | 73 | /** 74 | * Singleton that resolves Vehicle API client 75 | * @function getVehicleClient 76 | * @param {Promise} client - swagger client 77 | */ 78 | async function getVehicleClient () { 79 | if (!vehicleClient) { 80 | logger.debug('Vehicle Client resolve') 81 | vehicleClient = await SwaggerClient(`${vehicleAPIUri}/api-docs`) 82 | logger.debug('Vehicle Client succesfully resolved') 83 | } 84 | return vehicleClient 85 | } 86 | 87 | module.exports = { 88 | get 89 | } 90 | -------------------------------------------------------------------------------- /services/api-gateway/api/swagger/swagger.yaml: -------------------------------------------------------------------------------- 1 | swagger: "2.0" 2 | info: 3 | version: "0.0.1" 4 | title: API Gateway 5 | description: API Gateway 6 | basePath: / 7 | tags: 8 | - name: user 9 | description: User 10 | schemes: 11 | - http 12 | - https 13 | consumes: 14 | - application/json 15 | produces: 16 | - application/json 17 | paths: 18 | /users: 19 | x-swagger-router-controller: userController 20 | get: 21 | operationId: get 22 | tags: 23 | - user 24 | description: Return with users 25 | responses: 26 | "200": 27 | description: Success 28 | schema: 29 | type: array 30 | items: 31 | $ref: '#/definitions/User' 32 | default: 33 | description: Error 34 | schema: 35 | $ref: "#/definitions/ErrorResponse" 36 | /healthz: 37 | x-swagger-router-controller: healthzController 38 | get: 39 | description: Returns application health 40 | responses: 41 | "200": 42 | description: Success 43 | schema: 44 | type: object 45 | properties: 46 | status: 47 | type: string 48 | default: 49 | description: Error 50 | schema: 51 | $ref: "#/definitions/ErrorResponse" 52 | 53 | definitions: 54 | User: 55 | properties: 56 | id: 57 | type: string 58 | name: 59 | type: string 60 | vehicles: 61 | type: array 62 | items: 63 | $ref: '#/definitions/Vehicle' 64 | required: 65 | - id 66 | Vehicle: 67 | properties: 68 | id: 69 | type: string 70 | name: 71 | type: string 72 | required: 73 | - id 74 | ErrorResponse: 75 | properties: 76 | message: 77 | type: string 78 | required: 79 | - message 80 | -------------------------------------------------------------------------------- /services/api-gateway/app.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | const assert = require('assert') 4 | const path = require('path') 5 | const url = require('url') 6 | const SwaggerExpress = require('swagger-express-mw') 7 | const express = require('express') 8 | const logger = require('winston') 9 | const redis = require('redis') 10 | const Limiter = require('ratelimiter') 11 | 12 | assert(process.env.REDIS_URI, 'REDIS_URI env. variable is required') 13 | 14 | const app = express() 15 | const redisClient = redis.createClient(process.env.REDIS_URI) 16 | const swaggerUiPath = path.dirname(require.resolve('swagger-ui-dist')) 17 | const swaggerConfig = { 18 | appRoot: __dirname 19 | } 20 | 21 | const RATE_LIMITER_MAX = 50 22 | const RATE_LIMITER_DURATION_IN_MS = 60 * 1000 23 | 24 | SwaggerExpress.create(swaggerConfig, (err, swaggerExpress) => { 25 | if (err) { 26 | throw err 27 | } 28 | 29 | // Ratelimiter 30 | app.use((req, res, next) => { 31 | const limiter = new Limiter({ 32 | db: redisClient, 33 | id: (req) => req.connection.remoteAddress, 34 | max: RATE_LIMITER_MAX, 35 | duration: RATE_LIMITER_DURATION_IN_MS 36 | }) 37 | 38 | limiter.get((err, limit) => { 39 | if (err) { 40 | logger.error('Ratelimiter', err) 41 | next(err) 42 | return 43 | } 44 | 45 | res.set('RateLimit-Limit', limit.total) 46 | res.set('RateLimit-Remaining', limit.remaining - 1) 47 | res.set('RateLimit-Reset', limit.reset) 48 | 49 | // Too many requests 50 | if (limit.remaining < 1) { 51 | const after = Math.floor(limit.reset - (Date.now() / 1000)) 52 | 53 | logger.debug('Ratelimiter', limit) 54 | 55 | res.set('Retry-After', after) 56 | res.status(429).json({ 57 | message: 'Too Many Requests' 58 | }) 59 | return 60 | } 61 | 62 | next() 63 | }) 64 | }) 65 | 66 | // Swagger routes 67 | swaggerExpress.register(app) 68 | 69 | // API Docs 70 | app.get('/', (req, res) => { 71 | const query = req.query 72 | query.url = '/api-docs' 73 | res.redirect(301, url.format({ 74 | pathname: '/docs', 75 | query 76 | })) 77 | }) 78 | 79 | app.get('/api-docs', (req, res) => { 80 | res.set('Content-Type', 'text/yaml') 81 | res.sendFile(path.join(__dirname, 'api/swagger/swagger.yaml')) 82 | }) 83 | 84 | app.get('/docs', (req, res, next) => { 85 | if (!req.query.url) { 86 | const query = req.query 87 | query.url = '/api-docs' 88 | res.redirect(301, url.format({ query })) 89 | return 90 | } 91 | next() 92 | }) 93 | 94 | app.use('/docs', express.static(swaggerUiPath)) 95 | 96 | // Error handler 97 | app.use((err, req, res, next) => { 98 | if (err.message === 'Validation errors') { 99 | res.statusCode = err.statusCode 100 | res.json({ 101 | message: 'Error', 102 | errors: err.errors 103 | }) 104 | next() 105 | return 106 | } 107 | 108 | next(err) 109 | }) 110 | }) 111 | 112 | module.exports = app 113 | -------------------------------------------------------------------------------- /services/api-gateway/config/default.yaml: -------------------------------------------------------------------------------- 1 | # Default config from: https://github.com/swagger-api/swagger-node/blob/master/project-skeletons/connect/config/default.yaml 2 | swagger: 3 | fittingsDirs: [ api/fittings ] 4 | defaultPipe: null 5 | swaggerControllerPipe: swagger_controllers 6 | bagpipes: 7 | 8 | _router: 9 | name: swagger_router 10 | mockMode: false 11 | mockControllersDirs: [ api/mocks ] 12 | controllersDirs: [ api/controllers ] 13 | 14 | _swagger_validate: 15 | name: swagger_validator 16 | validateResponse: true 17 | 18 | swagger_controllers: 19 | - onError: json_error_handler 20 | - cors 21 | - swagger_params_parser 22 | - swagger_security 23 | - _swagger_validate 24 | - express_compatibility 25 | - _router 26 | 27 | swagger_raw: 28 | name: swagger_raw 29 | -------------------------------------------------------------------------------- /services/api-gateway/index.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | const jaeger = require('jaeger-client') 4 | const UDPSender = require('jaeger-client/dist/src/reporters/udp_sender').default 5 | const Instrument = require('@risingstack/opentracing-auto') 6 | 7 | const sampler = new jaeger.RateLimitingSampler(10) 8 | const reporter = new jaeger.RemoteReporter(new UDPSender({ 9 | host: process.env.JAEGER_HOST 10 | })) 11 | const tracer = new jaeger.Tracer('api-gateway', reporter, sampler) 12 | 13 | // eslint-disable-next-line 14 | new Instrument({ 15 | tracers: [tracer], 16 | httpTimings: true 17 | }) 18 | 19 | require('./server') 20 | 21 | process.on('unhandledRejection', (err, promise) => { 22 | console.error('An unhandledRejection occurred') 23 | console.error(err) 24 | console.error(`Rejected Promise: ${promise}`) 25 | console.error(`Rejection: ${err}`) 26 | process.exit(1) 27 | }) 28 | -------------------------------------------------------------------------------- /services/api-gateway/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "api-gateway", 3 | "version": "0.0.1", 4 | "private": true, 5 | "description": "API Gateway", 6 | "main": "index.js", 7 | "scripts": { 8 | "start": "node index.js" 9 | }, 10 | "dependencies": { 11 | "@risingstack/opentracing-auto": "1.4.3", 12 | "express": "4.15.4", 13 | "jaeger-client": "3.5.3", 14 | "ratelimiter": "3.0.3", 15 | "redis": "2.8.0", 16 | "swagger-client": "3.1.1", 17 | "swagger-express-mw": "0.7.0", 18 | "swagger-ui-dist": "3.1.5", 19 | "winston": "2.3.1" 20 | } 21 | } 22 | -------------------------------------------------------------------------------- /services/api-gateway/server.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | const logger = require('winston') 4 | const app = require('./app') 5 | 6 | const port = process.env.PORT || 3000 7 | const logLevel = process.env.LOG_LEVEL || 'debug' 8 | 9 | logger.level = logLevel 10 | 11 | const server = app.listen(port, (err) => { 12 | if (err) { 13 | logger.error(err) 14 | return 15 | } 16 | 17 | logger.info(`Server is listening on ${port}`) 18 | }) 19 | 20 | process.on('SIGTERM', () => { 21 | logger.info('SIGTERM received') 22 | 23 | server.close((err) => { 24 | if (err) { 25 | logger.error('Graceful shutdown', err) 26 | process.exit(1) 27 | } 28 | 29 | logger.info('Server stopped') 30 | process.exit(0) 31 | }) 32 | }) 33 | -------------------------------------------------------------------------------- /services/user-api/.dockerignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | -------------------------------------------------------------------------------- /services/user-api/Dockerfile: -------------------------------------------------------------------------------- 1 | FROM risingstack/alpine:3.4-v8.5.0-4.7.0 2 | 3 | ENV PORT 3000 4 | 5 | EXPOSE 3000 6 | 7 | COPY package.json package.json 8 | RUN npm install 9 | 10 | COPY . . 11 | 12 | CMD ["node", "."] 13 | -------------------------------------------------------------------------------- /services/user-api/api/controllers/healthzController.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | /** 4 | * List users via user-api and vehicle-api 5 | * @function get 6 | * @param {express.Request} req 7 | * @param {express.Response} res 8 | * @param {Function} next 9 | */ 10 | function get (req, res, next) { 11 | res.json({ 12 | status: 'ok' 13 | }) 14 | next() 15 | } 16 | 17 | module.exports = { 18 | get 19 | } 20 | -------------------------------------------------------------------------------- /services/user-api/api/controllers/userController.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | // Simple user database 4 | const USER_DB = [ 5 | { 6 | id: 1, 7 | name: 'Samantha' 8 | }, 9 | { 10 | id: 2, 11 | name: 'John' 12 | } 13 | ] 14 | 15 | /** 16 | * List users 17 | * @function get 18 | * @param {express.Request} req 19 | * @param {express.Response} res 20 | * @param {Function} next 21 | */ 22 | async function get (req, res, next) { 23 | res.json(USER_DB) 24 | next() 25 | } 26 | 27 | module.exports = { 28 | get 29 | } 30 | -------------------------------------------------------------------------------- /services/user-api/api/swagger/swagger.yaml: -------------------------------------------------------------------------------- 1 | swagger: "2.0" 2 | info: 3 | version: "0.0.1" 4 | title: User API 5 | description: API Gateway 6 | basePath: / 7 | tags: 8 | - name: user 9 | description: User API 10 | schemes: 11 | - http 12 | - https 13 | consumes: 14 | - application/json 15 | produces: 16 | - application/json 17 | paths: 18 | /users: 19 | x-swagger-router-controller: userController 20 | get: 21 | operationId: get 22 | tags: 23 | - user 24 | description: Return with users 25 | responses: 26 | "200": 27 | description: Success 28 | schema: 29 | type: array 30 | items: 31 | $ref: '#/definitions/User' 32 | default: 33 | description: Error 34 | schema: 35 | $ref: "#/definitions/ErrorResponse" 36 | /healthz: 37 | x-swagger-router-controller: healthzController 38 | get: 39 | description: Returns application health 40 | responses: 41 | "200": 42 | description: Success 43 | schema: 44 | type: object 45 | properties: 46 | status: 47 | type: string 48 | default: 49 | description: Error 50 | schema: 51 | $ref: "#/definitions/ErrorResponse" 52 | definitions: 53 | User: 54 | properties: 55 | id: 56 | type: string 57 | name: 58 | type: string 59 | required: 60 | - id 61 | ErrorResponse: 62 | properties: 63 | message: 64 | type: string 65 | required: 66 | - message 67 | -------------------------------------------------------------------------------- /services/user-api/app.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | const path = require('path') 4 | const url = require('url') 5 | const SwaggerExpress = require('swagger-express-mw') 6 | const express = require('express') 7 | 8 | const app = express() 9 | const swaggerUiPath = path.dirname(require.resolve('swagger-ui-dist')) 10 | const swaggerConfig = { 11 | appRoot: __dirname 12 | } 13 | 14 | SwaggerExpress.create(swaggerConfig, (err, swaggerExpress) => { 15 | if (err) { 16 | throw err 17 | } 18 | 19 | swaggerExpress.register(app) 20 | 21 | app.get('/api-docs', (req, res) => { 22 | res.set('Content-Type', 'text/yaml') 23 | res.sendFile(path.join(__dirname, 'api/swagger/swagger.yaml')) 24 | }) 25 | 26 | app.get('/docs', (req, res, next) => { 27 | if (!req.query.url) { 28 | const query = req.query 29 | query.url = '/api-docs' 30 | res.redirect(301, url.format({ query })) 31 | return 32 | } 33 | next() 34 | }) 35 | 36 | app.use('/docs', express.static(swaggerUiPath)) 37 | 38 | // Error handler 39 | app.use((err, req, res, next) => { 40 | if (err.message === 'Validation errors') { 41 | res.statusCode = err.statusCode 42 | res.json({ 43 | message: 'Error', 44 | errors: err.errors 45 | }) 46 | next() 47 | return 48 | } 49 | 50 | next(err) 51 | }) 52 | }) 53 | 54 | module.exports = app 55 | -------------------------------------------------------------------------------- /services/user-api/config/default.yaml: -------------------------------------------------------------------------------- 1 | # Default config from: https://github.com/swagger-api/swagger-node/blob/master/project-skeletons/connect/config/default.yaml 2 | swagger: 3 | fittingsDirs: [ api/fittings ] 4 | defaultPipe: null 5 | swaggerControllerPipe: swagger_controllers 6 | bagpipes: 7 | 8 | _router: 9 | name: swagger_router 10 | mockMode: false 11 | mockControllersDirs: [ api/mocks ] 12 | controllersDirs: [ api/controllers ] 13 | 14 | _swagger_validate: 15 | name: swagger_validator 16 | validateResponse: true 17 | 18 | swagger_controllers: 19 | - onError: json_error_handler 20 | - cors 21 | - swagger_params_parser 22 | - swagger_security 23 | - _swagger_validate 24 | - express_compatibility 25 | - _router 26 | 27 | swagger_raw: 28 | name: swagger_raw 29 | -------------------------------------------------------------------------------- /services/user-api/index.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | const jaeger = require('jaeger-client') 4 | const UDPSender = require('jaeger-client/dist/src/reporters/udp_sender').default 5 | const Instrument = require('@risingstack/opentracing-auto') 6 | 7 | const sampler = new jaeger.RateLimitingSampler(10) 8 | const reporter = new jaeger.RemoteReporter(new UDPSender({ 9 | host: process.env.JAEGER_HOST 10 | })) 11 | const tracer = new jaeger.Tracer('user-api', reporter, sampler) 12 | 13 | // eslint-disable-next-line 14 | new Instrument({ 15 | tracers: [tracer], 16 | httpTimings: true 17 | }) 18 | 19 | require('./server') 20 | 21 | process.on('unhandledRejection', (err, promise) => { 22 | console.error('An unhandledRejection occurred') 23 | console.error(err) 24 | console.error(`Rejected Promise: ${promise}`) 25 | console.error(`Rejection: ${err}`) 26 | process.exit(1) 27 | }) 28 | -------------------------------------------------------------------------------- /services/user-api/package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "user-api", 3 | "version": "0.0.1", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@risingstack/opentracing-auto": { 8 | "version": "1.4.3", 9 | "resolved": "https://registry.npmjs.org/@risingstack/opentracing-auto/-/opentracing-auto-1.4.3.tgz", 10 | "integrity": "sha1-OgyBGigOwJhn6j52EYJPKiH2wV4=", 11 | "requires": { 12 | "asyncctx": "1.0.1", 13 | "debug": "3.0.1", 14 | "lodash": "4.17.4", 15 | "methods": "1.1.2", 16 | "opentracing": "0.14.1", 17 | "require-in-the-middle": "2.1.2", 18 | "semver": "5.4.1", 19 | "shimmer": "1.1.0", 20 | "uuid": "3.1.0" 21 | }, 22 | "dependencies": { 23 | "debug": { 24 | "version": "3.0.1", 25 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.0.1.tgz", 26 | "integrity": "sha512-6nVc6S36qbt/mutyt+UGMnawAMrPDZUPQjRZI3FS9tCtDRhvxJbK79unYBLPi+z5SLXQ3ftoVBFCblQtNSls8w==", 27 | "requires": { 28 | "ms": "2.0.0" 29 | } 30 | }, 31 | "lodash": { 32 | "version": "4.17.4", 33 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.4.tgz", 34 | "integrity": "sha1-eCA6TRwyiuHYbcpkYONptX9AVa4=" 35 | } 36 | } 37 | }, 38 | "accepts": { 39 | "version": "1.3.4", 40 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.4.tgz", 41 | "integrity": "sha1-hiRnWMfdbSGmR0/whKR0DsBesh8=", 42 | "requires": { 43 | "mime-types": "2.1.17", 44 | "negotiator": "0.6.1" 45 | } 46 | }, 47 | "ajv": { 48 | "version": "4.11.8", 49 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-4.11.8.tgz", 50 | "integrity": "sha1-gv+wKynmYq5TvcIK8VlHcGc5xTY=", 51 | "requires": { 52 | "co": "4.6.0", 53 | "json-stable-stringify": "1.0.1" 54 | } 55 | }, 56 | "ansi-color": { 57 | "version": "0.2.1", 58 | "resolved": "https://registry.npmjs.org/ansi-color/-/ansi-color-0.2.1.tgz", 59 | "integrity": "sha1-PnXAN0dSF1RO12Oo21cJ+prlv5o=" 60 | }, 61 | "append-field": { 62 | "version": "0.1.0", 63 | "resolved": "https://registry.npmjs.org/append-field/-/append-field-0.1.0.tgz", 64 | "integrity": "sha1-bdxY+gg8e8VF08WZWygwzCNm1Eo=" 65 | }, 66 | "argparse": { 67 | "version": "1.0.9", 68 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.9.tgz", 69 | "integrity": "sha1-c9g7wmP4bpf4zE9rrhsOkKfSLIY=", 70 | "requires": { 71 | "sprintf-js": "1.0.3" 72 | } 73 | }, 74 | "array-flatten": { 75 | "version": "1.1.1", 76 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 77 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" 78 | }, 79 | "asn1": { 80 | "version": "0.2.3", 81 | "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.3.tgz", 82 | "integrity": "sha1-2sh4dxPJlmhJ/IGAd36+nB3fO4Y=" 83 | }, 84 | "assert-plus": { 85 | "version": "0.2.0", 86 | "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-0.2.0.tgz", 87 | "integrity": "sha1-104bh+ev/A24qttwIfP+SBAasjQ=" 88 | }, 89 | "async": { 90 | "version": "1.5.2", 91 | "resolved": "https://registry.npmjs.org/async/-/async-1.5.2.tgz", 92 | "integrity": "sha1-7GphrlZIDAw8skHJVhjiCJL5Zyo=" 93 | }, 94 | "asyncctx": { 95 | "version": "1.0.1", 96 | "resolved": "https://registry.npmjs.org/asyncctx/-/asyncctx-1.0.1.tgz", 97 | "integrity": "sha512-p7D8XqJusyrxKyeSSB7y28cppjAu3zcSw9sql9RtHwVnNPezlg3LvJoUFoawfzx/oEpLx1fy06AO5o9vfDR/TQ==", 98 | "requires": { 99 | "stack-chain": "1.3.7" 100 | } 101 | }, 102 | "asynckit": { 103 | "version": "0.4.0", 104 | "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", 105 | "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=" 106 | }, 107 | "aws-sign2": { 108 | "version": "0.6.0", 109 | "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.6.0.tgz", 110 | "integrity": "sha1-FDQt0428yU0OW4fXY81jYSwOeU8=" 111 | }, 112 | "aws4": { 113 | "version": "1.6.0", 114 | "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.6.0.tgz", 115 | "integrity": "sha1-g+9cqGCysy5KDe7e6MdxudtXRx4=" 116 | }, 117 | "bagpipes": { 118 | "version": "0.1.0", 119 | "resolved": "https://registry.npmjs.org/bagpipes/-/bagpipes-0.1.0.tgz", 120 | "integrity": "sha1-0oAFC0BEkozJw9uwMhweuKvQ1/A=", 121 | "requires": { 122 | "async": "1.5.2", 123 | "debug": "2.6.8", 124 | "jspath": "0.3.4", 125 | "lodash": "3.10.1", 126 | "machinepack-http": "2.4.0", 127 | "mustache": "2.3.0", 128 | "pipeworks": "1.3.1" 129 | } 130 | }, 131 | "bcrypt-pbkdf": { 132 | "version": "1.0.1", 133 | "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.1.tgz", 134 | "integrity": "sha1-Y7xdy2EzG5K8Bf1SiVPDNGKgb40=", 135 | "optional": true, 136 | "requires": { 137 | "tweetnacl": "0.14.5" 138 | } 139 | }, 140 | "body-parser": { 141 | "version": "1.18.1", 142 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.18.1.tgz", 143 | "integrity": "sha512-KL2pZpGvy6xuZHgYUznB1Zfw4AoGMApfRanT5NafeLvglbaSM+4CCtmlyYOv66oYXqvKL1xpaFb94V/AZVUnYg==", 144 | "requires": { 145 | "bytes": "3.0.0", 146 | "content-type": "1.0.4", 147 | "debug": "2.6.8", 148 | "depd": "1.1.1", 149 | "http-errors": "1.6.2", 150 | "iconv-lite": "0.4.19", 151 | "on-finished": "2.3.0", 152 | "qs": "6.5.1", 153 | "raw-body": "2.3.2", 154 | "type-is": "1.6.15" 155 | }, 156 | "dependencies": { 157 | "qs": { 158 | "version": "6.5.1", 159 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.1.tgz", 160 | "integrity": "sha512-eRzhrN1WSINYCDCbrz796z37LOe3m5tmW7RQf6oBntukAG1nmovJvhnwHHRMAfeoItc1m2Hk02WER2aQ/iqs+A==" 161 | } 162 | } 163 | }, 164 | "boom": { 165 | "version": "2.10.1", 166 | "resolved": "https://registry.npmjs.org/boom/-/boom-2.10.1.tgz", 167 | "integrity": "sha1-OciRjO/1eZ+D+UkqhI9iWt0Mdm8=", 168 | "requires": { 169 | "hoek": "2.16.3" 170 | } 171 | }, 172 | "bufrw": { 173 | "version": "1.2.1", 174 | "resolved": "https://registry.npmjs.org/bufrw/-/bufrw-1.2.1.tgz", 175 | "integrity": "sha1-k/IiIptPX14s1VkjaJFAf5hTZjs=", 176 | "requires": { 177 | "ansi-color": "0.2.1", 178 | "error": "7.0.2", 179 | "xtend": "4.0.1" 180 | } 181 | }, 182 | "busboy": { 183 | "version": "0.2.14", 184 | "resolved": "https://registry.npmjs.org/busboy/-/busboy-0.2.14.tgz", 185 | "integrity": "sha1-bCpiLvz0fFe7vh4qnDetNseSVFM=", 186 | "requires": { 187 | "dicer": "0.2.5", 188 | "readable-stream": "1.1.14" 189 | } 190 | }, 191 | "bytes": { 192 | "version": "3.0.0", 193 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz", 194 | "integrity": "sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg=" 195 | }, 196 | "caseless": { 197 | "version": "0.12.0", 198 | "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz", 199 | "integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=" 200 | }, 201 | "chance": { 202 | "version": "1.0.11", 203 | "resolved": "https://registry.npmjs.org/chance/-/chance-1.0.11.tgz", 204 | "integrity": "sha512-PK8Zmz6kJAkwwJuiTZvjcFX11/H+BvVd2r8dnHrDRXmOLoUHh4Hxiv+YyYTcT7sdFzP9LhMS3O3C/7f+7gSQ2g==" 205 | }, 206 | "co": { 207 | "version": "4.6.0", 208 | "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", 209 | "integrity": "sha1-bqa989hTrlTMuOR7+gvz+QMfsYQ=" 210 | }, 211 | "colors": { 212 | "version": "1.0.3", 213 | "resolved": "https://registry.npmjs.org/colors/-/colors-1.0.3.tgz", 214 | "integrity": "sha1-BDP0TYCWgP3rYO0mDxsMJi6CpAs=" 215 | }, 216 | "combined-stream": { 217 | "version": "1.0.5", 218 | "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.5.tgz", 219 | "integrity": "sha1-k4NwpXtKUd6ix3wV1cX9+JUWQAk=", 220 | "requires": { 221 | "delayed-stream": "1.0.0" 222 | } 223 | }, 224 | "commander": { 225 | "version": "2.11.0", 226 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.11.0.tgz", 227 | "integrity": "sha512-b0553uYA5YAEGgyYIGYROzKQ7X5RAqedkfjiZxwi0kL1g3bOaBNNZfYkzt/CL0umgD5wc9Jec2FbB98CjkMRvQ==" 228 | }, 229 | "component-emitter": { 230 | "version": "1.2.1", 231 | "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.2.1.tgz", 232 | "integrity": "sha1-E3kY1teCg/ffemt8WmPhQOaUJeY=" 233 | }, 234 | "concat-stream": { 235 | "version": "1.6.0", 236 | "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.0.tgz", 237 | "integrity": "sha1-CqxmL9Ur54lk1VMvaUeE5wEQrPc=", 238 | "requires": { 239 | "inherits": "2.0.3", 240 | "readable-stream": "2.3.3", 241 | "typedarray": "0.0.6" 242 | }, 243 | "dependencies": { 244 | "isarray": { 245 | "version": "1.0.0", 246 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 247 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 248 | }, 249 | "readable-stream": { 250 | "version": "2.3.3", 251 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.3.tgz", 252 | "integrity": "sha512-m+qzzcn7KUxEmd1gMbchF+Y2eIUbieUaxkWtptyHywrX0rE8QEYqPC07Vuy4Wm32/xE16NcdBctb8S0Xe/5IeQ==", 253 | "requires": { 254 | "core-util-is": "1.0.2", 255 | "inherits": "2.0.3", 256 | "isarray": "1.0.0", 257 | "process-nextick-args": "1.0.7", 258 | "safe-buffer": "5.1.1", 259 | "string_decoder": "1.0.3", 260 | "util-deprecate": "1.0.2" 261 | } 262 | }, 263 | "string_decoder": { 264 | "version": "1.0.3", 265 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.0.3.tgz", 266 | "integrity": "sha512-4AH6Z5fzNNBcH+6XDMfA/BTt87skxqJlO0lAh3Dker5zThcAxG6mKz+iGu308UKoPPQ8Dcqx/4JhujzltRa+hQ==", 267 | "requires": { 268 | "safe-buffer": "5.1.1" 269 | } 270 | } 271 | } 272 | }, 273 | "config": { 274 | "version": "1.26.2", 275 | "resolved": "https://registry.npmjs.org/config/-/config-1.26.2.tgz", 276 | "integrity": "sha1-JGYpEWjYr64Kroq5nqTUJy9SDK4=", 277 | "requires": { 278 | "json5": "0.4.0", 279 | "os-homedir": "1.0.2" 280 | } 281 | }, 282 | "content-disposition": { 283 | "version": "0.5.2", 284 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.2.tgz", 285 | "integrity": "sha1-DPaLud318r55YcOoUXjLhdunjLQ=" 286 | }, 287 | "content-type": { 288 | "version": "1.0.4", 289 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", 290 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" 291 | }, 292 | "convert-to-ecmascript-compatible-varname": { 293 | "version": "0.1.5", 294 | "resolved": "https://registry.npmjs.org/convert-to-ecmascript-compatible-varname/-/convert-to-ecmascript-compatible-varname-0.1.5.tgz", 295 | "integrity": "sha1-9npJOMUjNENWQlBHnGcBS6yHhJk=" 296 | }, 297 | "cookie": { 298 | "version": "0.3.1", 299 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.3.1.tgz", 300 | "integrity": "sha1-5+Ch+e9DtMi6klxcWpboBtFoc7s=" 301 | }, 302 | "cookie-signature": { 303 | "version": "1.0.6", 304 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 305 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" 306 | }, 307 | "cookiejar": { 308 | "version": "2.1.1", 309 | "resolved": "https://registry.npmjs.org/cookiejar/-/cookiejar-2.1.1.tgz", 310 | "integrity": "sha1-Qa1XsbVVlR7BcUEqgZQrHoIA00o=" 311 | }, 312 | "core-util-is": { 313 | "version": "1.0.2", 314 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", 315 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" 316 | }, 317 | "cors": { 318 | "version": "2.8.4", 319 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.4.tgz", 320 | "integrity": "sha1-K9OB8usgECAQXNUOpZ2mMJBpRoY=", 321 | "requires": { 322 | "object-assign": "4.1.1", 323 | "vary": "1.1.1" 324 | } 325 | }, 326 | "cryptiles": { 327 | "version": "2.0.5", 328 | "resolved": "https://registry.npmjs.org/cryptiles/-/cryptiles-2.0.5.tgz", 329 | "integrity": "sha1-O9/s3GCBR8HGcgL6KR59ylnqo7g=", 330 | "requires": { 331 | "boom": "2.10.1" 332 | } 333 | }, 334 | "cycle": { 335 | "version": "1.0.3", 336 | "resolved": "https://registry.npmjs.org/cycle/-/cycle-1.0.3.tgz", 337 | "integrity": "sha1-IegLK+hYD5i0aPN5QwZisEbDStI=" 338 | }, 339 | "dashdash": { 340 | "version": "1.14.1", 341 | "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", 342 | "integrity": "sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=", 343 | "requires": { 344 | "assert-plus": "1.0.0" 345 | }, 346 | "dependencies": { 347 | "assert-plus": { 348 | "version": "1.0.0", 349 | "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", 350 | "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=" 351 | } 352 | } 353 | }, 354 | "debug": { 355 | "version": "2.6.8", 356 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.8.tgz", 357 | "integrity": "sha1-5zFTHKLt4n0YgiJCfaF4IdaP9Pw=", 358 | "requires": { 359 | "ms": "2.0.0" 360 | } 361 | }, 362 | "deep-extend": { 363 | "version": "0.4.2", 364 | "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.4.2.tgz", 365 | "integrity": "sha1-SLaZwn4zS/ifEIkr5DL25MfTSn8=" 366 | }, 367 | "delayed-stream": { 368 | "version": "1.0.0", 369 | "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", 370 | "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=" 371 | }, 372 | "depd": { 373 | "version": "1.1.1", 374 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.1.tgz", 375 | "integrity": "sha1-V4O04cRZ8G+lyif5kfPQbnoxA1k=" 376 | }, 377 | "deref": { 378 | "version": "0.6.4", 379 | "resolved": "https://registry.npmjs.org/deref/-/deref-0.6.4.tgz", 380 | "integrity": "sha1-vVqW1F2+0wEbuBvfaN31S+jhvU4=", 381 | "requires": { 382 | "deep-extend": "0.4.2" 383 | } 384 | }, 385 | "destroy": { 386 | "version": "1.0.4", 387 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", 388 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" 389 | }, 390 | "dicer": { 391 | "version": "0.2.5", 392 | "resolved": "https://registry.npmjs.org/dicer/-/dicer-0.2.5.tgz", 393 | "integrity": "sha1-WZbAhrszIYyBLAkL3cCc0S+stw8=", 394 | "requires": { 395 | "readable-stream": "1.1.14", 396 | "streamsearch": "0.1.2" 397 | } 398 | }, 399 | "discontinuous-range": { 400 | "version": "1.0.0", 401 | "resolved": "https://registry.npmjs.org/discontinuous-range/-/discontinuous-range-1.0.0.tgz", 402 | "integrity": "sha1-44Mx8IRLukm5qctxx3FYWqsbxlo=" 403 | }, 404 | "ecc-jsbn": { 405 | "version": "0.1.1", 406 | "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.1.tgz", 407 | "integrity": "sha1-D8c6ntXw1Tw4GTOYUj735UN3dQU=", 408 | "optional": true, 409 | "requires": { 410 | "jsbn": "0.1.1" 411 | } 412 | }, 413 | "ee-first": { 414 | "version": "1.1.1", 415 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 416 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" 417 | }, 418 | "encodeurl": { 419 | "version": "1.0.1", 420 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.1.tgz", 421 | "integrity": "sha1-eePVhlU0aQn+bw9Fpd5oEDspTSA=" 422 | }, 423 | "error": { 424 | "version": "7.0.2", 425 | "resolved": "https://registry.npmjs.org/error/-/error-7.0.2.tgz", 426 | "integrity": "sha1-pfdf/02ZJhJt2sDqXcOOaJFTywI=", 427 | "requires": { 428 | "string-template": "0.2.1", 429 | "xtend": "4.0.1" 430 | } 431 | }, 432 | "escape-html": { 433 | "version": "1.0.3", 434 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 435 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" 436 | }, 437 | "esprima": { 438 | "version": "4.0.0", 439 | "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.0.tgz", 440 | "integrity": "sha512-oftTcaMu/EGrEIu904mWteKIv8vMuOgGYo7EhVJJN00R/EED9DCua/xxHRdYnKtcECzVg7xOWhflvJMnqcFZjw==" 441 | }, 442 | "etag": { 443 | "version": "1.8.1", 444 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 445 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" 446 | }, 447 | "express": { 448 | "version": "4.15.4", 449 | "resolved": "https://registry.npmjs.org/express/-/express-4.15.4.tgz", 450 | "integrity": "sha1-Ay4iU0ic+PzgJma+yj0R7XotrtE=", 451 | "requires": { 452 | "accepts": "1.3.4", 453 | "array-flatten": "1.1.1", 454 | "content-disposition": "0.5.2", 455 | "content-type": "1.0.4", 456 | "cookie": "0.3.1", 457 | "cookie-signature": "1.0.6", 458 | "debug": "2.6.8", 459 | "depd": "1.1.1", 460 | "encodeurl": "1.0.1", 461 | "escape-html": "1.0.3", 462 | "etag": "1.8.1", 463 | "finalhandler": "1.0.5", 464 | "fresh": "0.5.0", 465 | "merge-descriptors": "1.0.1", 466 | "methods": "1.1.2", 467 | "on-finished": "2.3.0", 468 | "parseurl": "1.3.2", 469 | "path-to-regexp": "0.1.7", 470 | "proxy-addr": "1.1.5", 471 | "qs": "6.5.0", 472 | "range-parser": "1.2.0", 473 | "send": "0.15.4", 474 | "serve-static": "1.12.4", 475 | "setprototypeof": "1.0.3", 476 | "statuses": "1.3.1", 477 | "type-is": "1.6.15", 478 | "utils-merge": "1.0.0", 479 | "vary": "1.1.1" 480 | } 481 | }, 482 | "extend": { 483 | "version": "3.0.1", 484 | "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.1.tgz", 485 | "integrity": "sha1-p1Xqe8Gt/MWjHOfnYtuq3F5jZEQ=" 486 | }, 487 | "extsprintf": { 488 | "version": "1.3.0", 489 | "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", 490 | "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=" 491 | }, 492 | "eyes": { 493 | "version": "0.1.8", 494 | "resolved": "https://registry.npmjs.org/eyes/-/eyes-0.1.8.tgz", 495 | "integrity": "sha1-Ys8SAjTGg3hdkCNIqADvPgzCC8A=" 496 | }, 497 | "faker": { 498 | "version": "3.1.0", 499 | "resolved": "https://registry.npmjs.org/faker/-/faker-3.1.0.tgz", 500 | "integrity": "sha1-D5CPr05uwCUk5UpX5DLFwBPgjJ8=" 501 | }, 502 | "finalhandler": { 503 | "version": "1.0.5", 504 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.0.5.tgz", 505 | "integrity": "sha1-pwEwPSV6G8gv6lR6M+WuiVMXI98=", 506 | "requires": { 507 | "debug": "2.6.8", 508 | "encodeurl": "1.0.1", 509 | "escape-html": "1.0.3", 510 | "on-finished": "2.3.0", 511 | "parseurl": "1.3.2", 512 | "statuses": "1.3.1", 513 | "unpipe": "1.0.0" 514 | } 515 | }, 516 | "forever-agent": { 517 | "version": "0.6.1", 518 | "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", 519 | "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=" 520 | }, 521 | "form-data": { 522 | "version": "2.1.4", 523 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.1.4.tgz", 524 | "integrity": "sha1-M8GDrPGTJ27KqYFDpp6Uv+4XUNE=", 525 | "requires": { 526 | "asynckit": "0.4.0", 527 | "combined-stream": "1.0.5", 528 | "mime-types": "2.1.17" 529 | } 530 | }, 531 | "formidable": { 532 | "version": "1.1.1", 533 | "resolved": "https://registry.npmjs.org/formidable/-/formidable-1.1.1.tgz", 534 | "integrity": "sha1-lriIb3w8NQi5Mta9cMTTqI818ak=" 535 | }, 536 | "forwarded": { 537 | "version": "0.1.2", 538 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", 539 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=" 540 | }, 541 | "fresh": { 542 | "version": "0.5.0", 543 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.0.tgz", 544 | "integrity": "sha1-9HTKXmqSRtb9jglTz6m5yAWvp44=" 545 | }, 546 | "getpass": { 547 | "version": "0.1.7", 548 | "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", 549 | "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=", 550 | "requires": { 551 | "assert-plus": "1.0.0" 552 | }, 553 | "dependencies": { 554 | "assert-plus": { 555 | "version": "1.0.0", 556 | "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", 557 | "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=" 558 | } 559 | } 560 | }, 561 | "graphlib": { 562 | "version": "2.1.1", 563 | "resolved": "https://registry.npmjs.org/graphlib/-/graphlib-2.1.1.tgz", 564 | "integrity": "sha1-QjUsUrovTQNctWbrkfc5X3bryVE=", 565 | "requires": { 566 | "lodash": "4.17.4" 567 | }, 568 | "dependencies": { 569 | "lodash": { 570 | "version": "4.17.4", 571 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.4.tgz", 572 | "integrity": "sha1-eCA6TRwyiuHYbcpkYONptX9AVa4=" 573 | } 574 | } 575 | }, 576 | "har-schema": { 577 | "version": "1.0.5", 578 | "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-1.0.5.tgz", 579 | "integrity": "sha1-0mMTX0MwfALGAq/I/pWXDAFRNp4=" 580 | }, 581 | "har-validator": { 582 | "version": "4.2.1", 583 | "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-4.2.1.tgz", 584 | "integrity": "sha1-M0gdDxu/9gDdID11gSpqX7oALio=", 585 | "requires": { 586 | "ajv": "4.11.8", 587 | "har-schema": "1.0.5" 588 | } 589 | }, 590 | "hawk": { 591 | "version": "3.1.3", 592 | "resolved": "https://registry.npmjs.org/hawk/-/hawk-3.1.3.tgz", 593 | "integrity": "sha1-B4REvXwWQLD+VA0sm3PVlnjo4cQ=", 594 | "requires": { 595 | "boom": "2.10.1", 596 | "cryptiles": "2.0.5", 597 | "hoek": "2.16.3", 598 | "sntp": "1.0.9" 599 | } 600 | }, 601 | "hoek": { 602 | "version": "2.16.3", 603 | "resolved": "https://registry.npmjs.org/hoek/-/hoek-2.16.3.tgz", 604 | "integrity": "sha1-ILt0A9POo5jpHcRxCo/xuCdKJe0=" 605 | }, 606 | "http-errors": { 607 | "version": "1.6.2", 608 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.2.tgz", 609 | "integrity": "sha1-CgAsyFcHGSp+eUbO7cERVfYOxzY=", 610 | "requires": { 611 | "depd": "1.1.1", 612 | "inherits": "2.0.3", 613 | "setprototypeof": "1.0.3", 614 | "statuses": "1.3.1" 615 | } 616 | }, 617 | "http-signature": { 618 | "version": "1.1.1", 619 | "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.1.1.tgz", 620 | "integrity": "sha1-33LiZwZs0Kxn+3at+OE0qPvPkb8=", 621 | "requires": { 622 | "assert-plus": "0.2.0", 623 | "jsprim": "1.4.1", 624 | "sshpk": "1.13.1" 625 | } 626 | }, 627 | "iconv-lite": { 628 | "version": "0.4.19", 629 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.19.tgz", 630 | "integrity": "sha512-oTZqweIP51xaGPI4uPa56/Pri/480R+mo7SeU+YETByQNhDG55ycFyNLIgta9vXhILrxXDmF7ZGhqZIcuN0gJQ==" 631 | }, 632 | "inherits": { 633 | "version": "2.0.3", 634 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 635 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" 636 | }, 637 | "ipaddr.js": { 638 | "version": "1.4.0", 639 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.4.0.tgz", 640 | "integrity": "sha1-KWrKh4qCGBbluF0KKFqZvP9FgvA=" 641 | }, 642 | "is-typedarray": { 643 | "version": "1.0.0", 644 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", 645 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=" 646 | }, 647 | "isarray": { 648 | "version": "0.0.1", 649 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz", 650 | "integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8=" 651 | }, 652 | "isstream": { 653 | "version": "0.1.2", 654 | "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz", 655 | "integrity": "sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo=" 656 | }, 657 | "jaeger-client": { 658 | "version": "3.5.3", 659 | "resolved": "https://registry.npmjs.org/jaeger-client/-/jaeger-client-3.5.3.tgz", 660 | "integrity": "sha1-MKOYlkLuPo6gbRDCufngRtQk2O0=", 661 | "requires": { 662 | "node-int64": "0.4.0", 663 | "opentracing": "0.13.0", 664 | "thriftrw": "3.11.1", 665 | "xorshift": "0.2.1" 666 | }, 667 | "dependencies": { 668 | "opentracing": { 669 | "version": "0.13.0", 670 | "resolved": "https://registry.npmjs.org/opentracing/-/opentracing-0.13.0.tgz", 671 | "integrity": "sha1-ajQUQvCdfYZrwR7QPeHjgo49aqs=" 672 | } 673 | } 674 | }, 675 | "js-base64": { 676 | "version": "2.3.2", 677 | "resolved": "https://registry.npmjs.org/js-base64/-/js-base64-2.3.2.tgz", 678 | "integrity": "sha512-Y2/+DnfJJXT1/FCwUebUhLWb3QihxiSC42+ctHLGogmW2jPY6LCapMdFZXRvVP2z6qyKW7s6qncE/9gSqZiArw==" 679 | }, 680 | "js-yaml": { 681 | "version": "3.10.0", 682 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.10.0.tgz", 683 | "integrity": "sha512-O2v52ffjLa9VeM43J4XocZE//WT9N0IiwDa3KSHH7Tu8CtH+1qM8SIZvnsTh6v+4yFy5KUY3BHUVwjpfAWsjIA==", 684 | "requires": { 685 | "argparse": "1.0.9", 686 | "esprima": "4.0.0" 687 | } 688 | }, 689 | "jsbn": { 690 | "version": "0.1.1", 691 | "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", 692 | "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=", 693 | "optional": true 694 | }, 695 | "json-refs": { 696 | "version": "2.1.7", 697 | "resolved": "https://registry.npmjs.org/json-refs/-/json-refs-2.1.7.tgz", 698 | "integrity": "sha1-uesB/in16j6Sh48VrqEK04taz4k=", 699 | "requires": { 700 | "commander": "2.11.0", 701 | "graphlib": "2.1.1", 702 | "js-yaml": "3.10.0", 703 | "native-promise-only": "0.8.1", 704 | "path-loader": "1.0.2", 705 | "slash": "1.0.0", 706 | "uri-js": "3.0.2" 707 | } 708 | }, 709 | "json-schema": { 710 | "version": "0.2.3", 711 | "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.2.3.tgz", 712 | "integrity": "sha1-tIDIkuWaLwWVTOcnvT8qTogvnhM=" 713 | }, 714 | "json-schema-faker": { 715 | "version": "0.2.16", 716 | "resolved": "https://registry.npmjs.org/json-schema-faker/-/json-schema-faker-0.2.16.tgz", 717 | "integrity": "sha1-UdPKSJVdj+c09ZHXR7ckU75aePI=", 718 | "requires": { 719 | "chance": "1.0.11", 720 | "deref": "0.6.4", 721 | "faker": "3.1.0", 722 | "randexp": "0.4.6" 723 | } 724 | }, 725 | "json-stable-stringify": { 726 | "version": "1.0.1", 727 | "resolved": "https://registry.npmjs.org/json-stable-stringify/-/json-stable-stringify-1.0.1.tgz", 728 | "integrity": "sha1-mnWdOcXy/1A/1TAGRu1EX4jE+a8=", 729 | "requires": { 730 | "jsonify": "0.0.0" 731 | } 732 | }, 733 | "json-stringify-safe": { 734 | "version": "5.0.1", 735 | "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", 736 | "integrity": "sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=" 737 | }, 738 | "json5": { 739 | "version": "0.4.0", 740 | "resolved": "https://registry.npmjs.org/json5/-/json5-0.4.0.tgz", 741 | "integrity": "sha1-BUNS5MTIDIbAkjh31EneF2pzLI0=" 742 | }, 743 | "jsonify": { 744 | "version": "0.0.0", 745 | "resolved": "https://registry.npmjs.org/jsonify/-/jsonify-0.0.0.tgz", 746 | "integrity": "sha1-LHS27kHZPKUbe1qu6PUDYx0lKnM=" 747 | }, 748 | "jspath": { 749 | "version": "0.3.4", 750 | "resolved": "https://registry.npmjs.org/jspath/-/jspath-0.3.4.tgz", 751 | "integrity": "sha1-2J0+0uh0NP5s0ASyQskS35aXNSQ=" 752 | }, 753 | "jsprim": { 754 | "version": "1.4.1", 755 | "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.1.tgz", 756 | "integrity": "sha1-MT5mvB5cwG5Di8G3SZwuXFastqI=", 757 | "requires": { 758 | "assert-plus": "1.0.0", 759 | "extsprintf": "1.3.0", 760 | "json-schema": "0.2.3", 761 | "verror": "1.10.0" 762 | }, 763 | "dependencies": { 764 | "assert-plus": { 765 | "version": "1.0.0", 766 | "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", 767 | "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=" 768 | } 769 | } 770 | }, 771 | "lodash": { 772 | "version": "3.10.1", 773 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-3.10.1.tgz", 774 | "integrity": "sha1-W/Rejkm6QYnhfUgnid/RW9FAt7Y=" 775 | }, 776 | "lodash.get": { 777 | "version": "4.4.2", 778 | "resolved": "https://registry.npmjs.org/lodash.get/-/lodash.get-4.4.2.tgz", 779 | "integrity": "sha1-LRd/ZS+jHpObRDjVNBSZ36OCXpk=" 780 | }, 781 | "lodash.isequal": { 782 | "version": "4.5.0", 783 | "resolved": "https://registry.npmjs.org/lodash.isequal/-/lodash.isequal-4.5.0.tgz", 784 | "integrity": "sha1-QVxEePK8wwEgwizhDtMib30+GOA=" 785 | }, 786 | "long": { 787 | "version": "2.4.0", 788 | "resolved": "https://registry.npmjs.org/long/-/long-2.4.0.tgz", 789 | "integrity": "sha1-n6GAux2VAM3CnEFWdmoZleH0Uk8=" 790 | }, 791 | "machine": { 792 | "version": "10.4.0", 793 | "resolved": "https://registry.npmjs.org/machine/-/machine-10.4.0.tgz", 794 | "integrity": "sha1-m1Ys5GeCEzKCijd9GQ65NrTkB7I=", 795 | "requires": { 796 | "convert-to-ecmascript-compatible-varname": "0.1.5", 797 | "debug": "2.6.8", 798 | "lodash": "3.10.1", 799 | "object-hash": "0.3.0", 800 | "rttc": "7.4.0", 801 | "switchback": "2.0.2" 802 | } 803 | }, 804 | "machinepack-http": { 805 | "version": "2.4.0", 806 | "resolved": "https://registry.npmjs.org/machinepack-http/-/machinepack-http-2.4.0.tgz", 807 | "integrity": "sha1-CnhcF9xrnBuaxAiBvu+uiudIVek=", 808 | "requires": { 809 | "lodash": "3.10.1", 810 | "machine": "10.4.0", 811 | "machinepack-urls": "4.1.0", 812 | "request": "2.81.0" 813 | } 814 | }, 815 | "machinepack-urls": { 816 | "version": "4.1.0", 817 | "resolved": "https://registry.npmjs.org/machinepack-urls/-/machinepack-urls-4.1.0.tgz", 818 | "integrity": "sha1-0l4y6Xw8LLiVaLqMmNIp1cMF45E=", 819 | "requires": { 820 | "lodash": "3.10.1", 821 | "machine": "9.1.2" 822 | }, 823 | "dependencies": { 824 | "machine": { 825 | "version": "9.1.2", 826 | "resolved": "https://registry.npmjs.org/machine/-/machine-9.1.2.tgz", 827 | "integrity": "sha1-hL+Pt3ZqlqplqpbWbpUJ62oFqDQ=", 828 | "requires": { 829 | "convert-to-ecmascript-compatible-varname": "0.1.5", 830 | "debug": "2.6.8", 831 | "lodash": "3.10.1", 832 | "object-hash": "0.3.0", 833 | "rttc": "4.5.2", 834 | "switchback": "1.1.3" 835 | } 836 | }, 837 | "rttc": { 838 | "version": "4.5.2", 839 | "resolved": "https://registry.npmjs.org/rttc/-/rttc-4.5.2.tgz", 840 | "integrity": "sha1-umo+komLQnTxI7usSUhddhajfLw=", 841 | "requires": { 842 | "lodash": "3.10.1" 843 | } 844 | }, 845 | "switchback": { 846 | "version": "1.1.3", 847 | "resolved": "https://registry.npmjs.org/switchback/-/switchback-1.1.3.tgz", 848 | "integrity": "sha1-EscBCTSNailvc5upEO64U/i25jE=", 849 | "requires": { 850 | "lodash": "2.4.2" 851 | }, 852 | "dependencies": { 853 | "lodash": { 854 | "version": "2.4.2", 855 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-2.4.2.tgz", 856 | "integrity": "sha1-+t2DS5aDBz2hebPq5tnA0VBT9z4=" 857 | } 858 | } 859 | } 860 | } 861 | }, 862 | "media-typer": { 863 | "version": "0.3.0", 864 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 865 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" 866 | }, 867 | "merge-descriptors": { 868 | "version": "1.0.1", 869 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 870 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" 871 | }, 872 | "methods": { 873 | "version": "1.1.2", 874 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 875 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" 876 | }, 877 | "mime": { 878 | "version": "1.3.4", 879 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.3.4.tgz", 880 | "integrity": "sha1-EV+eO2s9rylZmDyzjxSaLUDrXVM=" 881 | }, 882 | "mime-db": { 883 | "version": "1.30.0", 884 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.30.0.tgz", 885 | "integrity": "sha1-dMZD2i3Z1qRTmZY0ZbJtXKfXHwE=" 886 | }, 887 | "mime-types": { 888 | "version": "2.1.17", 889 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.17.tgz", 890 | "integrity": "sha1-Cdejk/A+mVp5+K+Fe3Cp4KsWVXo=", 891 | "requires": { 892 | "mime-db": "1.30.0" 893 | } 894 | }, 895 | "minimist": { 896 | "version": "0.0.8", 897 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", 898 | "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=" 899 | }, 900 | "mkdirp": { 901 | "version": "0.5.1", 902 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", 903 | "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", 904 | "requires": { 905 | "minimist": "0.0.8" 906 | } 907 | }, 908 | "module-details-from-path": { 909 | "version": "1.0.3", 910 | "resolved": "https://registry.npmjs.org/module-details-from-path/-/module-details-from-path-1.0.3.tgz", 911 | "integrity": "sha1-EUyUlnPiqKNenTV4hSeqN7Z52is=" 912 | }, 913 | "ms": { 914 | "version": "2.0.0", 915 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 916 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 917 | }, 918 | "multer": { 919 | "version": "1.3.0", 920 | "resolved": "https://registry.npmjs.org/multer/-/multer-1.3.0.tgz", 921 | "integrity": "sha1-CSsmcPaEb6SRSWXvyM+Uwg/sbNI=", 922 | "requires": { 923 | "append-field": "0.1.0", 924 | "busboy": "0.2.14", 925 | "concat-stream": "1.6.0", 926 | "mkdirp": "0.5.1", 927 | "object-assign": "3.0.0", 928 | "on-finished": "2.3.0", 929 | "type-is": "1.6.15", 930 | "xtend": "4.0.1" 931 | }, 932 | "dependencies": { 933 | "object-assign": { 934 | "version": "3.0.0", 935 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-3.0.0.tgz", 936 | "integrity": "sha1-m+3VygiXlJvKR+f/QIBi1Un1h/I=" 937 | } 938 | } 939 | }, 940 | "mustache": { 941 | "version": "2.3.0", 942 | "resolved": "https://registry.npmjs.org/mustache/-/mustache-2.3.0.tgz", 943 | "integrity": "sha1-QCj3d4sXcIpImTCm5SrDvKDaQdA=" 944 | }, 945 | "native-promise-only": { 946 | "version": "0.8.1", 947 | "resolved": "https://registry.npmjs.org/native-promise-only/-/native-promise-only-0.8.1.tgz", 948 | "integrity": "sha1-IKMYwwy0X3H+et+/eyHJnBRy7xE=" 949 | }, 950 | "negotiator": { 951 | "version": "0.6.1", 952 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.1.tgz", 953 | "integrity": "sha1-KzJxhOiZIQEXeyhWP7XnECrNDKk=" 954 | }, 955 | "node-int64": { 956 | "version": "0.4.0", 957 | "resolved": "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz", 958 | "integrity": "sha1-h6kGXNs1XTGC2PlM4RGIuCXGijs=" 959 | }, 960 | "oauth-sign": { 961 | "version": "0.8.2", 962 | "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.8.2.tgz", 963 | "integrity": "sha1-Rqarfwrq2N6unsBWV4C31O/rnUM=" 964 | }, 965 | "object-assign": { 966 | "version": "4.1.1", 967 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 968 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=" 969 | }, 970 | "object-hash": { 971 | "version": "0.3.0", 972 | "resolved": "https://registry.npmjs.org/object-hash/-/object-hash-0.3.0.tgz", 973 | "integrity": "sha1-VIII5Ds2pE5NowutbFasU7iF50Q=" 974 | }, 975 | "on-finished": { 976 | "version": "2.3.0", 977 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", 978 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", 979 | "requires": { 980 | "ee-first": "1.1.1" 981 | } 982 | }, 983 | "opentracing": { 984 | "version": "0.14.1", 985 | "resolved": "https://registry.npmjs.org/opentracing/-/opentracing-0.14.1.tgz", 986 | "integrity": "sha1-QNJ4vupBdmCjXdnT7nZRH/qRHc0=" 987 | }, 988 | "os-homedir": { 989 | "version": "1.0.2", 990 | "resolved": "https://registry.npmjs.org/os-homedir/-/os-homedir-1.0.2.tgz", 991 | "integrity": "sha1-/7xJiDNuDoM94MFox+8VISGqf7M=" 992 | }, 993 | "parseurl": { 994 | "version": "1.3.2", 995 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.2.tgz", 996 | "integrity": "sha1-/CidTtiZMRlGDBViUyYs3I3mW/M=" 997 | }, 998 | "path-loader": { 999 | "version": "1.0.2", 1000 | "resolved": "https://registry.npmjs.org/path-loader/-/path-loader-1.0.2.tgz", 1001 | "integrity": "sha1-zVxz5+OakQEb4UjWv92KhbuTHvk=", 1002 | "requires": { 1003 | "native-promise-only": "0.8.1", 1004 | "superagent": "3.6.0" 1005 | } 1006 | }, 1007 | "path-parse": { 1008 | "version": "1.0.5", 1009 | "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.5.tgz", 1010 | "integrity": "sha1-PBrfhx6pzWyUMbbqK9dKD/BVxME=" 1011 | }, 1012 | "path-to-regexp": { 1013 | "version": "0.1.7", 1014 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 1015 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" 1016 | }, 1017 | "performance-now": { 1018 | "version": "0.2.0", 1019 | "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-0.2.0.tgz", 1020 | "integrity": "sha1-M+8wxcd9TqIcWlOGnZG1bY8lVeU=" 1021 | }, 1022 | "pipeworks": { 1023 | "version": "1.3.1", 1024 | "resolved": "https://registry.npmjs.org/pipeworks/-/pipeworks-1.3.1.tgz", 1025 | "integrity": "sha1-+ENvhWXtHZe/OoBjKlOXv9NTOF8=" 1026 | }, 1027 | "process-nextick-args": { 1028 | "version": "1.0.7", 1029 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-1.0.7.tgz", 1030 | "integrity": "sha1-FQ4gt1ZZCtP5EJPyWk8q2L/zC6M=" 1031 | }, 1032 | "proxy-addr": { 1033 | "version": "1.1.5", 1034 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-1.1.5.tgz", 1035 | "integrity": "sha1-ccDuOxAt4/IC87ZPYI0XP8uhqRg=", 1036 | "requires": { 1037 | "forwarded": "0.1.2", 1038 | "ipaddr.js": "1.4.0" 1039 | } 1040 | }, 1041 | "punycode": { 1042 | "version": "1.4.1", 1043 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz", 1044 | "integrity": "sha1-wNWmOycYgArY4esPpSachN1BhF4=" 1045 | }, 1046 | "qs": { 1047 | "version": "6.5.0", 1048 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.0.tgz", 1049 | "integrity": "sha512-fjVFjW9yhqMhVGwRExCXLhJKrLlkYSaxNWdyc9rmHlrVZbk35YHH312dFd7191uQeXkI3mKLZTIbSvIeFwFemg==" 1050 | }, 1051 | "randexp": { 1052 | "version": "0.4.6", 1053 | "resolved": "https://registry.npmjs.org/randexp/-/randexp-0.4.6.tgz", 1054 | "integrity": "sha512-80WNmd9DA0tmZrw9qQa62GPPWfuXJknrmVmLcxvq4uZBdYqb1wYoKTmnlGUchvVWe0XiLupYkBoXVOxz3C8DYQ==", 1055 | "requires": { 1056 | "discontinuous-range": "1.0.0", 1057 | "ret": "0.1.15" 1058 | } 1059 | }, 1060 | "range-parser": { 1061 | "version": "1.2.0", 1062 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.0.tgz", 1063 | "integrity": "sha1-9JvmtIeJTdxA3MlKMi9hEJLgDV4=" 1064 | }, 1065 | "raw-body": { 1066 | "version": "2.3.2", 1067 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.3.2.tgz", 1068 | "integrity": "sha1-vNYMd9Prk83gBQKVw/N5OJvIj4k=", 1069 | "requires": { 1070 | "bytes": "3.0.0", 1071 | "http-errors": "1.6.2", 1072 | "iconv-lite": "0.4.19", 1073 | "unpipe": "1.0.0" 1074 | } 1075 | }, 1076 | "readable-stream": { 1077 | "version": "1.1.14", 1078 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.1.14.tgz", 1079 | "integrity": "sha1-fPTFTvZI44EwhMY23SB54WbAgdk=", 1080 | "requires": { 1081 | "core-util-is": "1.0.2", 1082 | "inherits": "2.0.3", 1083 | "isarray": "0.0.1", 1084 | "string_decoder": "0.10.31" 1085 | } 1086 | }, 1087 | "request": { 1088 | "version": "2.81.0", 1089 | "resolved": "https://registry.npmjs.org/request/-/request-2.81.0.tgz", 1090 | "integrity": "sha1-xpKJRqDgbF+Nb4qTM0af/aRimKA=", 1091 | "requires": { 1092 | "aws-sign2": "0.6.0", 1093 | "aws4": "1.6.0", 1094 | "caseless": "0.12.0", 1095 | "combined-stream": "1.0.5", 1096 | "extend": "3.0.1", 1097 | "forever-agent": "0.6.1", 1098 | "form-data": "2.1.4", 1099 | "har-validator": "4.2.1", 1100 | "hawk": "3.1.3", 1101 | "http-signature": "1.1.1", 1102 | "is-typedarray": "1.0.0", 1103 | "isstream": "0.1.2", 1104 | "json-stringify-safe": "5.0.1", 1105 | "mime-types": "2.1.17", 1106 | "oauth-sign": "0.8.2", 1107 | "performance-now": "0.2.0", 1108 | "qs": "6.4.0", 1109 | "safe-buffer": "5.1.1", 1110 | "stringstream": "0.0.5", 1111 | "tough-cookie": "2.3.2", 1112 | "tunnel-agent": "0.6.0", 1113 | "uuid": "3.1.0" 1114 | }, 1115 | "dependencies": { 1116 | "qs": { 1117 | "version": "6.4.0", 1118 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.4.0.tgz", 1119 | "integrity": "sha1-E+JtKK1rD/qpExLNO/cI7TUecjM=" 1120 | } 1121 | } 1122 | }, 1123 | "require-in-the-middle": { 1124 | "version": "2.1.2", 1125 | "resolved": "https://registry.npmjs.org/require-in-the-middle/-/require-in-the-middle-2.1.2.tgz", 1126 | "integrity": "sha1-vduJMW1FvNsI4sYYa9Lm6Bmo7q4=", 1127 | "requires": { 1128 | "module-details-from-path": "1.0.3", 1129 | "resolve": "1.4.0" 1130 | } 1131 | }, 1132 | "resolve": { 1133 | "version": "1.4.0", 1134 | "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.4.0.tgz", 1135 | "integrity": "sha512-aW7sVKPufyHqOmyyLzg/J+8606v5nevBgaliIlV7nUpVMsDnoBGV/cbSLNjZAg9q0Cfd/+easKVKQ8vOu8fn1Q==", 1136 | "requires": { 1137 | "path-parse": "1.0.5" 1138 | } 1139 | }, 1140 | "ret": { 1141 | "version": "0.1.15", 1142 | "resolved": "https://registry.npmjs.org/ret/-/ret-0.1.15.tgz", 1143 | "integrity": "sha512-TTlYpa+OL+vMMNG24xSlQGEJ3B/RzEfUlLct7b5G/ytav+wPrplCpVMFuwzXbkecJrb6IYo1iFb0S9v37754mg==" 1144 | }, 1145 | "rttc": { 1146 | "version": "7.4.0", 1147 | "resolved": "https://registry.npmjs.org/rttc/-/rttc-7.4.0.tgz", 1148 | "integrity": "sha1-vJys1Grdkj3rYklaAZNOt+9hn7Q=", 1149 | "requires": { 1150 | "lodash": "3.10.1" 1151 | } 1152 | }, 1153 | "safe-buffer": { 1154 | "version": "5.1.1", 1155 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.1.tgz", 1156 | "integrity": "sha512-kKvNJn6Mm93gAczWVJg7wH+wGYWNrDHdWvpUmHyEsgCtIwwo3bqPtV4tR5tuPaUhTOo/kvhVwd8XwwOllGYkbg==" 1157 | }, 1158 | "semver": { 1159 | "version": "5.4.1", 1160 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.4.1.tgz", 1161 | "integrity": "sha512-WfG/X9+oATh81XtllIo/I8gOiY9EXRdv1cQdyykeXK17YcUW3EXUAi2To4pcH6nZtJPr7ZOpM5OMyWJZm+8Rsg==" 1162 | }, 1163 | "send": { 1164 | "version": "0.15.4", 1165 | "resolved": "https://registry.npmjs.org/send/-/send-0.15.4.tgz", 1166 | "integrity": "sha1-mF+qPihLAnPHkzZKNcZze9k5Bbk=", 1167 | "requires": { 1168 | "debug": "2.6.8", 1169 | "depd": "1.1.1", 1170 | "destroy": "1.0.4", 1171 | "encodeurl": "1.0.1", 1172 | "escape-html": "1.0.3", 1173 | "etag": "1.8.1", 1174 | "fresh": "0.5.0", 1175 | "http-errors": "1.6.2", 1176 | "mime": "1.3.4", 1177 | "ms": "2.0.0", 1178 | "on-finished": "2.3.0", 1179 | "range-parser": "1.2.0", 1180 | "statuses": "1.3.1" 1181 | } 1182 | }, 1183 | "serve-static": { 1184 | "version": "1.12.4", 1185 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.12.4.tgz", 1186 | "integrity": "sha1-m2qpjutyU8Tu3Ewfb9vKYJkBqWE=", 1187 | "requires": { 1188 | "encodeurl": "1.0.1", 1189 | "escape-html": "1.0.3", 1190 | "parseurl": "1.3.2", 1191 | "send": "0.15.4" 1192 | } 1193 | }, 1194 | "setprototypeof": { 1195 | "version": "1.0.3", 1196 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.0.3.tgz", 1197 | "integrity": "sha1-ZlZ+NwQ+608E2RvWWMDL77VbjgQ=" 1198 | }, 1199 | "shimmer": { 1200 | "version": "1.1.0", 1201 | "resolved": "https://registry.npmjs.org/shimmer/-/shimmer-1.1.0.tgz", 1202 | "integrity": "sha1-l9c3cTf/u6tCVSLkKf4KqJpIizU=" 1203 | }, 1204 | "slash": { 1205 | "version": "1.0.0", 1206 | "resolved": "https://registry.npmjs.org/slash/-/slash-1.0.0.tgz", 1207 | "integrity": "sha1-xB8vbDn8FtHNF61LXYlhFK5HDVU=" 1208 | }, 1209 | "sntp": { 1210 | "version": "1.0.9", 1211 | "resolved": "https://registry.npmjs.org/sntp/-/sntp-1.0.9.tgz", 1212 | "integrity": "sha1-ZUEYTMkK7qbG57NeJlkIJEPGYZg=", 1213 | "requires": { 1214 | "hoek": "2.16.3" 1215 | } 1216 | }, 1217 | "sprintf-js": { 1218 | "version": "1.0.3", 1219 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", 1220 | "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=" 1221 | }, 1222 | "sshpk": { 1223 | "version": "1.13.1", 1224 | "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.13.1.tgz", 1225 | "integrity": "sha1-US322mKHFEMW3EwY/hzx2UBzm+M=", 1226 | "requires": { 1227 | "asn1": "0.2.3", 1228 | "assert-plus": "1.0.0", 1229 | "bcrypt-pbkdf": "1.0.1", 1230 | "dashdash": "1.14.1", 1231 | "ecc-jsbn": "0.1.1", 1232 | "getpass": "0.1.7", 1233 | "jsbn": "0.1.1", 1234 | "tweetnacl": "0.14.5" 1235 | }, 1236 | "dependencies": { 1237 | "assert-plus": { 1238 | "version": "1.0.0", 1239 | "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", 1240 | "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=" 1241 | } 1242 | } 1243 | }, 1244 | "stack-chain": { 1245 | "version": "1.3.7", 1246 | "resolved": "https://registry.npmjs.org/stack-chain/-/stack-chain-1.3.7.tgz", 1247 | "integrity": "sha1-0ZLJ/06moiyUxN1FkXHj8AzqEoU=" 1248 | }, 1249 | "stack-trace": { 1250 | "version": "0.0.10", 1251 | "resolved": "https://registry.npmjs.org/stack-trace/-/stack-trace-0.0.10.tgz", 1252 | "integrity": "sha1-VHxws0fo0ytOEI6hoqFZ5f3eGcA=" 1253 | }, 1254 | "statuses": { 1255 | "version": "1.3.1", 1256 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.3.1.tgz", 1257 | "integrity": "sha1-+vUbnrdKrvOzrPStX2Gr8ky3uT4=" 1258 | }, 1259 | "streamsearch": { 1260 | "version": "0.1.2", 1261 | "resolved": "https://registry.npmjs.org/streamsearch/-/streamsearch-0.1.2.tgz", 1262 | "integrity": "sha1-gIudDlb8Jz2Am6VzOOkpkZoanxo=" 1263 | }, 1264 | "string_decoder": { 1265 | "version": "0.10.31", 1266 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-0.10.31.tgz", 1267 | "integrity": "sha1-YuIDvEF2bGwoyfyEMB2rHFMQ+pQ=" 1268 | }, 1269 | "string-template": { 1270 | "version": "0.2.1", 1271 | "resolved": "https://registry.npmjs.org/string-template/-/string-template-0.2.1.tgz", 1272 | "integrity": "sha1-QpMuWYo1LQH8IuwzZ9nYTuxsmt0=" 1273 | }, 1274 | "stringstream": { 1275 | "version": "0.0.5", 1276 | "resolved": "https://registry.npmjs.org/stringstream/-/stringstream-0.0.5.tgz", 1277 | "integrity": "sha1-TkhM1N5aC7vuGORjB3EKioFiGHg=" 1278 | }, 1279 | "superagent": { 1280 | "version": "3.6.0", 1281 | "resolved": "https://registry.npmjs.org/superagent/-/superagent-3.6.0.tgz", 1282 | "integrity": "sha512-oWsu4mboo8sVxagp4bNwZIR1rUmypeAJDmNIwT9mF4k06hSu6P92aOjEWLaIj7vsX3fOUp+cRH/04tao+q5Q7A==", 1283 | "requires": { 1284 | "component-emitter": "1.2.1", 1285 | "cookiejar": "2.1.1", 1286 | "debug": "2.6.8", 1287 | "extend": "3.0.1", 1288 | "form-data": "2.1.4", 1289 | "formidable": "1.1.1", 1290 | "methods": "1.1.2", 1291 | "mime": "1.4.0", 1292 | "qs": "6.5.0", 1293 | "readable-stream": "2.3.3" 1294 | }, 1295 | "dependencies": { 1296 | "isarray": { 1297 | "version": "1.0.0", 1298 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 1299 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 1300 | }, 1301 | "mime": { 1302 | "version": "1.4.0", 1303 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.4.0.tgz", 1304 | "integrity": "sha512-n9ChLv77+QQEapYz8lV+rIZAW3HhAPW2CXnzb1GN5uMkuczshwvkW7XPsbzU0ZQN3sP47Er2KVkp2p3KyqZKSQ==" 1305 | }, 1306 | "readable-stream": { 1307 | "version": "2.3.3", 1308 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.3.tgz", 1309 | "integrity": "sha512-m+qzzcn7KUxEmd1gMbchF+Y2eIUbieUaxkWtptyHywrX0rE8QEYqPC07Vuy4Wm32/xE16NcdBctb8S0Xe/5IeQ==", 1310 | "requires": { 1311 | "core-util-is": "1.0.2", 1312 | "inherits": "2.0.3", 1313 | "isarray": "1.0.0", 1314 | "process-nextick-args": "1.0.7", 1315 | "safe-buffer": "5.1.1", 1316 | "string_decoder": "1.0.3", 1317 | "util-deprecate": "1.0.2" 1318 | } 1319 | }, 1320 | "string_decoder": { 1321 | "version": "1.0.3", 1322 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.0.3.tgz", 1323 | "integrity": "sha512-4AH6Z5fzNNBcH+6XDMfA/BTt87skxqJlO0lAh3Dker5zThcAxG6mKz+iGu308UKoPPQ8Dcqx/4JhujzltRa+hQ==", 1324 | "requires": { 1325 | "safe-buffer": "5.1.1" 1326 | } 1327 | } 1328 | } 1329 | }, 1330 | "swagger-express-mw": { 1331 | "version": "0.7.0", 1332 | "resolved": "https://registry.npmjs.org/swagger-express-mw/-/swagger-express-mw-0.7.0.tgz", 1333 | "integrity": "sha1-SfXbctHUs4JzNu5sx7Npyu1b9Mg=", 1334 | "requires": { 1335 | "swagger-node-runner": "0.7.3" 1336 | } 1337 | }, 1338 | "swagger-methods": { 1339 | "version": "1.0.0", 1340 | "resolved": "https://registry.npmjs.org/swagger-methods/-/swagger-methods-1.0.0.tgz", 1341 | "integrity": "sha1-s5x3lX0wWmU1wKHgFQgRhbmdYfw=" 1342 | }, 1343 | "swagger-node-runner": { 1344 | "version": "0.7.3", 1345 | "resolved": "https://registry.npmjs.org/swagger-node-runner/-/swagger-node-runner-0.7.3.tgz", 1346 | "integrity": "sha1-P0RH+ma8Mv9Kmm+qw8rVVnst3Go=", 1347 | "requires": { 1348 | "async": "1.5.2", 1349 | "bagpipes": "0.1.0", 1350 | "body-parser": "1.18.1", 1351 | "config": "1.26.2", 1352 | "cors": "2.8.4", 1353 | "debug": "2.6.8", 1354 | "js-yaml": "3.10.0", 1355 | "lodash": "3.10.1", 1356 | "multer": "1.3.0", 1357 | "parseurl": "1.3.2", 1358 | "qs": "6.5.0", 1359 | "sway": "1.0.0", 1360 | "type-is": "1.6.15" 1361 | } 1362 | }, 1363 | "swagger-schema-official": { 1364 | "version": "2.0.0-bab6bed", 1365 | "resolved": "https://registry.npmjs.org/swagger-schema-official/-/swagger-schema-official-2.0.0-bab6bed.tgz", 1366 | "integrity": "sha1-cAcEaNbSl3ylI3suUZyn0Gouo/0=" 1367 | }, 1368 | "swagger-ui-dist": { 1369 | "version": "3.1.5", 1370 | "resolved": "https://registry.npmjs.org/swagger-ui-dist/-/swagger-ui-dist-3.1.5.tgz", 1371 | "integrity": "sha1-R9JKQC97k10SBTBr9epEch9FnMo=" 1372 | }, 1373 | "sway": { 1374 | "version": "1.0.0", 1375 | "resolved": "https://registry.npmjs.org/sway/-/sway-1.0.0.tgz", 1376 | "integrity": "sha1-No/8Dpa9hCJu0bmzPWa+V9oE8Jo=", 1377 | "requires": { 1378 | "debug": "2.6.8", 1379 | "js-base64": "2.3.2", 1380 | "js-yaml": "3.10.0", 1381 | "json-refs": "2.1.7", 1382 | "json-schema-faker": "0.2.16", 1383 | "lodash": "4.17.4", 1384 | "native-promise-only": "0.8.1", 1385 | "path-to-regexp": "1.7.0", 1386 | "swagger-methods": "1.0.0", 1387 | "swagger-schema-official": "2.0.0-bab6bed", 1388 | "z-schema": "3.18.4" 1389 | }, 1390 | "dependencies": { 1391 | "lodash": { 1392 | "version": "4.17.4", 1393 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.4.tgz", 1394 | "integrity": "sha1-eCA6TRwyiuHYbcpkYONptX9AVa4=" 1395 | }, 1396 | "path-to-regexp": { 1397 | "version": "1.7.0", 1398 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-1.7.0.tgz", 1399 | "integrity": "sha1-Wf3g9DW62suhA6hOnTvGTpa5k30=", 1400 | "requires": { 1401 | "isarray": "0.0.1" 1402 | } 1403 | } 1404 | } 1405 | }, 1406 | "switchback": { 1407 | "version": "2.0.2", 1408 | "resolved": "https://registry.npmjs.org/switchback/-/switchback-2.0.2.tgz", 1409 | "integrity": "sha1-ls8ODTY7VZ0Lt/8htip6qRDsYHk=", 1410 | "requires": { 1411 | "lodash": "3.10.1" 1412 | } 1413 | }, 1414 | "thriftrw": { 1415 | "version": "3.11.1", 1416 | "resolved": "https://registry.npmjs.org/thriftrw/-/thriftrw-3.11.1.tgz", 1417 | "integrity": "sha1-Wi9RZdZluxleZl5bW5+Il9rCOsw=", 1418 | "requires": { 1419 | "bufrw": "1.2.1", 1420 | "error": "7.0.2", 1421 | "long": "2.4.0" 1422 | } 1423 | }, 1424 | "tough-cookie": { 1425 | "version": "2.3.2", 1426 | "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.3.2.tgz", 1427 | "integrity": "sha1-8IH3bkyFcg5sN6X6ztc3FQ2EByo=", 1428 | "requires": { 1429 | "punycode": "1.4.1" 1430 | } 1431 | }, 1432 | "tunnel-agent": { 1433 | "version": "0.6.0", 1434 | "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", 1435 | "integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=", 1436 | "requires": { 1437 | "safe-buffer": "5.1.1" 1438 | } 1439 | }, 1440 | "tweetnacl": { 1441 | "version": "0.14.5", 1442 | "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz", 1443 | "integrity": "sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q=", 1444 | "optional": true 1445 | }, 1446 | "type-is": { 1447 | "version": "1.6.15", 1448 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.15.tgz", 1449 | "integrity": "sha1-yrEPtJCeRByChC6v4a1kbIGARBA=", 1450 | "requires": { 1451 | "media-typer": "0.3.0", 1452 | "mime-types": "2.1.17" 1453 | } 1454 | }, 1455 | "typedarray": { 1456 | "version": "0.0.6", 1457 | "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", 1458 | "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=" 1459 | }, 1460 | "unpipe": { 1461 | "version": "1.0.0", 1462 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 1463 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" 1464 | }, 1465 | "uri-js": { 1466 | "version": "3.0.2", 1467 | "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-3.0.2.tgz", 1468 | "integrity": "sha1-+QuFhQf4HepNz7s8TD2/orVX+qo=", 1469 | "requires": { 1470 | "punycode": "2.1.0" 1471 | }, 1472 | "dependencies": { 1473 | "punycode": { 1474 | "version": "2.1.0", 1475 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.0.tgz", 1476 | "integrity": "sha1-X4Y+3Im5bbCQdLrXlHvwkFbKTn0=" 1477 | } 1478 | } 1479 | }, 1480 | "util-deprecate": { 1481 | "version": "1.0.2", 1482 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", 1483 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" 1484 | }, 1485 | "utils-merge": { 1486 | "version": "1.0.0", 1487 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.0.tgz", 1488 | "integrity": "sha1-ApT7kiu5N1FTVBxPcJYjHyh8ivg=" 1489 | }, 1490 | "uuid": { 1491 | "version": "3.1.0", 1492 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.1.0.tgz", 1493 | "integrity": "sha512-DIWtzUkw04M4k3bf1IcpS2tngXEL26YUD2M0tMDUpnUrz2hgzUBlD55a4FjdLGPvfHxS6uluGWvaVEqgBcVa+g==" 1494 | }, 1495 | "validator": { 1496 | "version": "8.2.0", 1497 | "resolved": "https://registry.npmjs.org/validator/-/validator-8.2.0.tgz", 1498 | "integrity": "sha512-Yw5wW34fSv5spzTXNkokD6S6/Oq92d8q/t14TqsS3fAiA1RYnxSFSIZ+CY3n6PGGRCq5HhJTSepQvFUS2QUDxA==" 1499 | }, 1500 | "vary": { 1501 | "version": "1.1.1", 1502 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.1.tgz", 1503 | "integrity": "sha1-Z1Neu2lMHVIldFeYRmUyP1h+jTc=" 1504 | }, 1505 | "verror": { 1506 | "version": "1.10.0", 1507 | "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz", 1508 | "integrity": "sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA=", 1509 | "requires": { 1510 | "assert-plus": "1.0.0", 1511 | "core-util-is": "1.0.2", 1512 | "extsprintf": "1.3.0" 1513 | }, 1514 | "dependencies": { 1515 | "assert-plus": { 1516 | "version": "1.0.0", 1517 | "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", 1518 | "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=" 1519 | } 1520 | } 1521 | }, 1522 | "winston": { 1523 | "version": "2.3.1", 1524 | "resolved": "https://registry.npmjs.org/winston/-/winston-2.3.1.tgz", 1525 | "integrity": "sha1-C0hCDZeMAYBM8CMLZIhhWYIloRk=", 1526 | "requires": { 1527 | "async": "1.0.0", 1528 | "colors": "1.0.3", 1529 | "cycle": "1.0.3", 1530 | "eyes": "0.1.8", 1531 | "isstream": "0.1.2", 1532 | "stack-trace": "0.0.10" 1533 | }, 1534 | "dependencies": { 1535 | "async": { 1536 | "version": "1.0.0", 1537 | "resolved": "https://registry.npmjs.org/async/-/async-1.0.0.tgz", 1538 | "integrity": "sha1-+PwEyjoTeErenhZBr5hXjPvWR6k=" 1539 | } 1540 | } 1541 | }, 1542 | "xorshift": { 1543 | "version": "0.2.1", 1544 | "resolved": "https://registry.npmjs.org/xorshift/-/xorshift-0.2.1.tgz", 1545 | "integrity": "sha1-/NgiZ+k1HBPw+5xzMH8lMx0pxjo=" 1546 | }, 1547 | "xtend": { 1548 | "version": "4.0.1", 1549 | "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.1.tgz", 1550 | "integrity": "sha1-pcbVMr5lbiPbgg77lDofBJmNY68=" 1551 | }, 1552 | "z-schema": { 1553 | "version": "3.18.4", 1554 | "resolved": "https://registry.npmjs.org/z-schema/-/z-schema-3.18.4.tgz", 1555 | "integrity": "sha512-DUOKC/IhbkdLKKiV89gw9DUauTV8U/8yJl1sjf6MtDmzevLKOF2duNJ495S3MFVjqZarr+qNGCPbkg4mu4PpLw==", 1556 | "requires": { 1557 | "commander": "2.11.0", 1558 | "lodash.get": "4.4.2", 1559 | "lodash.isequal": "4.5.0", 1560 | "validator": "8.2.0" 1561 | } 1562 | } 1563 | } 1564 | } 1565 | -------------------------------------------------------------------------------- /services/user-api/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "user-api", 3 | "version": "0.0.1", 4 | "private": true, 5 | "description": "User API", 6 | "main": "index.js", 7 | "scripts": { 8 | "start": "node index.js" 9 | }, 10 | "dependencies": { 11 | "@risingstack/opentracing-auto": "1.4.3", 12 | "express": "4.15.4", 13 | "jaeger-client": "3.5.3", 14 | "swagger-express-mw": "0.7.0", 15 | "swagger-ui-dist": "3.1.5", 16 | "winston": "2.3.1" 17 | } 18 | } 19 | -------------------------------------------------------------------------------- /services/user-api/server.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | const logger = require('winston') 4 | const app = require('./app') 5 | 6 | const port = process.env.PORT || 3000 7 | const logLevel = process.env.LOG_LEVEL || 'debug' 8 | 9 | logger.level = logLevel 10 | 11 | const server = app.listen(port, (err) => { 12 | if (err) { 13 | logger.error(err) 14 | return 15 | } 16 | 17 | logger.info(`Server is listening on ${port}`) 18 | }) 19 | 20 | process.on('SIGTERM', () => { 21 | logger.info('SIGTERM received') 22 | 23 | server.close((err) => { 24 | if (err) { 25 | logger.error('Graceful shutdown', err) 26 | process.exit(1) 27 | } 28 | 29 | logger.info('Server stopped') 30 | process.exit(0) 31 | }) 32 | }) 33 | -------------------------------------------------------------------------------- /services/vehicle-api/.dockerignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | -------------------------------------------------------------------------------- /services/vehicle-api/Dockerfile: -------------------------------------------------------------------------------- 1 | FROM risingstack/alpine:3.4-v8.5.0-4.7.0 2 | 3 | ENV PORT 3000 4 | 5 | EXPOSE 3000 6 | 7 | COPY package.json package.json 8 | RUN npm install 9 | 10 | COPY . . 11 | 12 | CMD ["node", "."] 13 | -------------------------------------------------------------------------------- /services/vehicle-api/api/controllers/healthzController.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | /** 4 | * List users via user-api and vehicle-api 5 | * @function get 6 | * @param {express.Request} req 7 | * @param {express.Response} res 8 | * @param {Function} next 9 | */ 10 | function get (req, res, next) { 11 | res.json({ 12 | status: 'ok' 13 | }) 14 | next() 15 | } 16 | 17 | module.exports = { 18 | get 19 | } 20 | -------------------------------------------------------------------------------- /services/vehicle-api/api/controllers/vehicleController.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | // Simple vehicle database 4 | const VEHICLE_DB = [ 5 | { 6 | id: 1, 7 | userId: 1, 8 | name: 'Toyota Corolla' 9 | }, 10 | { 11 | id: 2, 12 | userId: 1, 13 | name: 'Porsche 911' 14 | }, 15 | { 16 | id: 3, 17 | userId: 2, 18 | name: 'Ford Focus' 19 | } 20 | ] 21 | 22 | /** 23 | * List vehicles 24 | * @function get 25 | * @param {express.Request} req 26 | * @param {express.Response} res 27 | * @param {Function} next 28 | */ 29 | async function get (req, res, next) { 30 | // Filter vehicles by user ids 31 | const userIds = req.swagger.params.userIds.value 32 | const vehicles = userIds 33 | ? VEHICLE_DB.filter((vehicle) => userIds.includes(vehicle.userId)) 34 | : VEHICLE_DB 35 | 36 | res.json(vehicles) 37 | next() 38 | } 39 | 40 | module.exports = { 41 | get 42 | } 43 | -------------------------------------------------------------------------------- /services/vehicle-api/api/swagger/swagger.yaml: -------------------------------------------------------------------------------- 1 | swagger: "2.0" 2 | info: 3 | version: "0.0.1" 4 | title: Vehicle API 5 | description: API Gateway 6 | basePath: / 7 | tags: 8 | - name: vehicle 9 | description: Vehicle API 10 | schemes: 11 | - http 12 | - https 13 | consumes: 14 | - application/json 15 | produces: 16 | - application/json 17 | paths: 18 | /vehicles: 19 | x-swagger-router-controller: vehicleController 20 | get: 21 | operationId: get 22 | tags: 23 | - vehicle 24 | description: Return with vehicles 25 | parameters: 26 | - name: userIds 27 | description: User id(s) 28 | in: query 29 | type: array 30 | uniqueItems: true 31 | items: 32 | type: integer 33 | responses: 34 | "200": 35 | description: Success 36 | schema: 37 | type: array 38 | items: 39 | $ref: '#/definitions/Vehicle' 40 | default: 41 | description: Error 42 | schema: 43 | $ref: "#/definitions/ErrorResponse" 44 | /healthz: 45 | x-swagger-router-controller: healthzController 46 | get: 47 | description: Returns application health 48 | responses: 49 | "200": 50 | description: Success 51 | schema: 52 | type: object 53 | properties: 54 | status: 55 | type: string 56 | default: 57 | description: Error 58 | schema: 59 | $ref: "#/definitions/ErrorResponse" 60 | definitions: 61 | Vehicle: 62 | properties: 63 | id: 64 | type: string 65 | vehicleId: 66 | type: string 67 | name: 68 | type: string 69 | required: 70 | - id 71 | - vehicleId 72 | ErrorResponse: 73 | properties: 74 | message: 75 | type: string 76 | required: 77 | - message 78 | -------------------------------------------------------------------------------- /services/vehicle-api/app.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | const path = require('path') 4 | const url = require('url') 5 | const SwaggerExpress = require('swagger-express-mw') 6 | const express = require('express') 7 | 8 | const app = express() 9 | const swaggerUiPath = path.dirname(require.resolve('swagger-ui-dist')) 10 | const swaggerConfig = { 11 | appRoot: __dirname 12 | } 13 | 14 | SwaggerExpress.create(swaggerConfig, (err, swaggerExpress) => { 15 | if (err) { 16 | throw err 17 | } 18 | 19 | swaggerExpress.register(app) 20 | 21 | app.get('/api-docs', (req, res) => { 22 | res.set('Content-Type', 'text/yaml') 23 | res.sendFile(path.join(__dirname, 'api/swagger/swagger.yaml')) 24 | }) 25 | 26 | app.get('/docs', (req, res, next) => { 27 | if (!req.query.url) { 28 | const query = req.query 29 | query.url = '/api-docs' 30 | res.redirect(301, url.format({ query })) 31 | return 32 | } 33 | next() 34 | }) 35 | 36 | app.use('/docs', express.static(swaggerUiPath)) 37 | 38 | // Error handler 39 | app.use((err, req, res, next) => { 40 | if (err.message === 'Validation errors') { 41 | res.statusCode = err.statusCode 42 | res.json({ 43 | message: 'Error', 44 | errors: err.errors 45 | }) 46 | next() 47 | return 48 | } 49 | 50 | next(err) 51 | }) 52 | }) 53 | 54 | module.exports = app 55 | -------------------------------------------------------------------------------- /services/vehicle-api/config/default.yaml: -------------------------------------------------------------------------------- 1 | # Default config from: https://github.com/swagger-api/swagger-node/blob/master/project-skeletons/connect/config/default.yaml 2 | swagger: 3 | fittingsDirs: [ api/fittings ] 4 | defaultPipe: null 5 | swaggerControllerPipe: swagger_controllers 6 | bagpipes: 7 | 8 | _router: 9 | name: swagger_router 10 | mockMode: false 11 | mockControllersDirs: [ api/mocks ] 12 | controllersDirs: [ api/controllers ] 13 | 14 | _swagger_validate: 15 | name: swagger_validator 16 | validateResponse: true 17 | 18 | swagger_controllers: 19 | - onError: json_error_handler 20 | - cors 21 | - swagger_params_parser 22 | - swagger_security 23 | - _swagger_validate 24 | - express_compatibility 25 | - _router 26 | 27 | swagger_raw: 28 | name: swagger_raw 29 | -------------------------------------------------------------------------------- /services/vehicle-api/index.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | const jaeger = require('jaeger-client') 4 | const UDPSender = require('jaeger-client/dist/src/reporters/udp_sender').default 5 | const Instrument = require('@risingstack/opentracing-auto') 6 | 7 | const sampler = new jaeger.RateLimitingSampler(10) 8 | const reporter = new jaeger.RemoteReporter(new UDPSender({ 9 | host: process.env.JAEGER_HOST 10 | })) 11 | const tracer = new jaeger.Tracer('vehicle-api', reporter, sampler) 12 | 13 | // eslint-disable-next-line 14 | new Instrument({ 15 | tracers: [tracer], 16 | httpTimings: true 17 | }) 18 | 19 | require('./server') 20 | 21 | process.on('unhandledRejection', (err, promise) => { 22 | console.error('An unhandledRejection occurred') 23 | console.error(err) 24 | console.error(`Rejected Promise: ${promise}`) 25 | console.error(`Rejection: ${err}`) 26 | process.exit(1) 27 | }) 28 | -------------------------------------------------------------------------------- /services/vehicle-api/package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "user-api", 3 | "version": "0.0.1", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@risingstack/opentracing-auto": { 8 | "version": "1.4.3", 9 | "resolved": "https://registry.npmjs.org/@risingstack/opentracing-auto/-/opentracing-auto-1.4.3.tgz", 10 | "integrity": "sha1-OgyBGigOwJhn6j52EYJPKiH2wV4=", 11 | "requires": { 12 | "asyncctx": "1.0.1", 13 | "debug": "3.0.1", 14 | "lodash": "4.17.4", 15 | "methods": "1.1.2", 16 | "opentracing": "0.14.1", 17 | "require-in-the-middle": "2.1.2", 18 | "semver": "5.4.1", 19 | "shimmer": "1.1.0", 20 | "uuid": "3.1.0" 21 | }, 22 | "dependencies": { 23 | "debug": { 24 | "version": "3.0.1", 25 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.0.1.tgz", 26 | "integrity": "sha512-6nVc6S36qbt/mutyt+UGMnawAMrPDZUPQjRZI3FS9tCtDRhvxJbK79unYBLPi+z5SLXQ3ftoVBFCblQtNSls8w==", 27 | "requires": { 28 | "ms": "2.0.0" 29 | } 30 | }, 31 | "lodash": { 32 | "version": "4.17.4", 33 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.4.tgz", 34 | "integrity": "sha1-eCA6TRwyiuHYbcpkYONptX9AVa4=" 35 | } 36 | } 37 | }, 38 | "accepts": { 39 | "version": "1.3.4", 40 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.4.tgz", 41 | "integrity": "sha1-hiRnWMfdbSGmR0/whKR0DsBesh8=", 42 | "requires": { 43 | "mime-types": "2.1.17", 44 | "negotiator": "0.6.1" 45 | } 46 | }, 47 | "ajv": { 48 | "version": "4.11.8", 49 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-4.11.8.tgz", 50 | "integrity": "sha1-gv+wKynmYq5TvcIK8VlHcGc5xTY=", 51 | "requires": { 52 | "co": "4.6.0", 53 | "json-stable-stringify": "1.0.1" 54 | } 55 | }, 56 | "ansi-color": { 57 | "version": "0.2.1", 58 | "resolved": "https://registry.npmjs.org/ansi-color/-/ansi-color-0.2.1.tgz", 59 | "integrity": "sha1-PnXAN0dSF1RO12Oo21cJ+prlv5o=" 60 | }, 61 | "append-field": { 62 | "version": "0.1.0", 63 | "resolved": "https://registry.npmjs.org/append-field/-/append-field-0.1.0.tgz", 64 | "integrity": "sha1-bdxY+gg8e8VF08WZWygwzCNm1Eo=" 65 | }, 66 | "argparse": { 67 | "version": "1.0.9", 68 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.9.tgz", 69 | "integrity": "sha1-c9g7wmP4bpf4zE9rrhsOkKfSLIY=", 70 | "requires": { 71 | "sprintf-js": "1.0.3" 72 | } 73 | }, 74 | "array-flatten": { 75 | "version": "1.1.1", 76 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 77 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" 78 | }, 79 | "asn1": { 80 | "version": "0.2.3", 81 | "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.3.tgz", 82 | "integrity": "sha1-2sh4dxPJlmhJ/IGAd36+nB3fO4Y=" 83 | }, 84 | "assert-plus": { 85 | "version": "0.2.0", 86 | "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-0.2.0.tgz", 87 | "integrity": "sha1-104bh+ev/A24qttwIfP+SBAasjQ=" 88 | }, 89 | "async": { 90 | "version": "1.5.2", 91 | "resolved": "https://registry.npmjs.org/async/-/async-1.5.2.tgz", 92 | "integrity": "sha1-7GphrlZIDAw8skHJVhjiCJL5Zyo=" 93 | }, 94 | "asyncctx": { 95 | "version": "1.0.1", 96 | "resolved": "https://registry.npmjs.org/asyncctx/-/asyncctx-1.0.1.tgz", 97 | "integrity": "sha512-p7D8XqJusyrxKyeSSB7y28cppjAu3zcSw9sql9RtHwVnNPezlg3LvJoUFoawfzx/oEpLx1fy06AO5o9vfDR/TQ==", 98 | "requires": { 99 | "stack-chain": "1.3.7" 100 | } 101 | }, 102 | "asynckit": { 103 | "version": "0.4.0", 104 | "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", 105 | "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=" 106 | }, 107 | "aws-sign2": { 108 | "version": "0.6.0", 109 | "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.6.0.tgz", 110 | "integrity": "sha1-FDQt0428yU0OW4fXY81jYSwOeU8=" 111 | }, 112 | "aws4": { 113 | "version": "1.6.0", 114 | "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.6.0.tgz", 115 | "integrity": "sha1-g+9cqGCysy5KDe7e6MdxudtXRx4=" 116 | }, 117 | "bagpipes": { 118 | "version": "0.1.0", 119 | "resolved": "https://registry.npmjs.org/bagpipes/-/bagpipes-0.1.0.tgz", 120 | "integrity": "sha1-0oAFC0BEkozJw9uwMhweuKvQ1/A=", 121 | "requires": { 122 | "async": "1.5.2", 123 | "debug": "2.6.8", 124 | "jspath": "0.3.4", 125 | "lodash": "3.10.1", 126 | "machinepack-http": "2.4.0", 127 | "mustache": "2.3.0", 128 | "pipeworks": "1.3.1" 129 | } 130 | }, 131 | "bcrypt-pbkdf": { 132 | "version": "1.0.1", 133 | "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.1.tgz", 134 | "integrity": "sha1-Y7xdy2EzG5K8Bf1SiVPDNGKgb40=", 135 | "optional": true, 136 | "requires": { 137 | "tweetnacl": "0.14.5" 138 | } 139 | }, 140 | "body-parser": { 141 | "version": "1.18.1", 142 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.18.1.tgz", 143 | "integrity": "sha512-KL2pZpGvy6xuZHgYUznB1Zfw4AoGMApfRanT5NafeLvglbaSM+4CCtmlyYOv66oYXqvKL1xpaFb94V/AZVUnYg==", 144 | "requires": { 145 | "bytes": "3.0.0", 146 | "content-type": "1.0.4", 147 | "debug": "2.6.8", 148 | "depd": "1.1.1", 149 | "http-errors": "1.6.2", 150 | "iconv-lite": "0.4.19", 151 | "on-finished": "2.3.0", 152 | "qs": "6.5.1", 153 | "raw-body": "2.3.2", 154 | "type-is": "1.6.15" 155 | }, 156 | "dependencies": { 157 | "qs": { 158 | "version": "6.5.1", 159 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.1.tgz", 160 | "integrity": "sha512-eRzhrN1WSINYCDCbrz796z37LOe3m5tmW7RQf6oBntukAG1nmovJvhnwHHRMAfeoItc1m2Hk02WER2aQ/iqs+A==" 161 | } 162 | } 163 | }, 164 | "boom": { 165 | "version": "2.10.1", 166 | "resolved": "https://registry.npmjs.org/boom/-/boom-2.10.1.tgz", 167 | "integrity": "sha1-OciRjO/1eZ+D+UkqhI9iWt0Mdm8=", 168 | "requires": { 169 | "hoek": "2.16.3" 170 | } 171 | }, 172 | "bufrw": { 173 | "version": "1.2.1", 174 | "resolved": "https://registry.npmjs.org/bufrw/-/bufrw-1.2.1.tgz", 175 | "integrity": "sha1-k/IiIptPX14s1VkjaJFAf5hTZjs=", 176 | "requires": { 177 | "ansi-color": "0.2.1", 178 | "error": "7.0.2", 179 | "xtend": "4.0.1" 180 | } 181 | }, 182 | "busboy": { 183 | "version": "0.2.14", 184 | "resolved": "https://registry.npmjs.org/busboy/-/busboy-0.2.14.tgz", 185 | "integrity": "sha1-bCpiLvz0fFe7vh4qnDetNseSVFM=", 186 | "requires": { 187 | "dicer": "0.2.5", 188 | "readable-stream": "1.1.14" 189 | } 190 | }, 191 | "bytes": { 192 | "version": "3.0.0", 193 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz", 194 | "integrity": "sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg=" 195 | }, 196 | "caseless": { 197 | "version": "0.12.0", 198 | "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz", 199 | "integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=" 200 | }, 201 | "chance": { 202 | "version": "1.0.11", 203 | "resolved": "https://registry.npmjs.org/chance/-/chance-1.0.11.tgz", 204 | "integrity": "sha512-PK8Zmz6kJAkwwJuiTZvjcFX11/H+BvVd2r8dnHrDRXmOLoUHh4Hxiv+YyYTcT7sdFzP9LhMS3O3C/7f+7gSQ2g==" 205 | }, 206 | "co": { 207 | "version": "4.6.0", 208 | "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", 209 | "integrity": "sha1-bqa989hTrlTMuOR7+gvz+QMfsYQ=" 210 | }, 211 | "colors": { 212 | "version": "1.0.3", 213 | "resolved": "https://registry.npmjs.org/colors/-/colors-1.0.3.tgz", 214 | "integrity": "sha1-BDP0TYCWgP3rYO0mDxsMJi6CpAs=" 215 | }, 216 | "combined-stream": { 217 | "version": "1.0.5", 218 | "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.5.tgz", 219 | "integrity": "sha1-k4NwpXtKUd6ix3wV1cX9+JUWQAk=", 220 | "requires": { 221 | "delayed-stream": "1.0.0" 222 | } 223 | }, 224 | "commander": { 225 | "version": "2.11.0", 226 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.11.0.tgz", 227 | "integrity": "sha512-b0553uYA5YAEGgyYIGYROzKQ7X5RAqedkfjiZxwi0kL1g3bOaBNNZfYkzt/CL0umgD5wc9Jec2FbB98CjkMRvQ==" 228 | }, 229 | "component-emitter": { 230 | "version": "1.2.1", 231 | "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.2.1.tgz", 232 | "integrity": "sha1-E3kY1teCg/ffemt8WmPhQOaUJeY=" 233 | }, 234 | "concat-stream": { 235 | "version": "1.6.0", 236 | "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.0.tgz", 237 | "integrity": "sha1-CqxmL9Ur54lk1VMvaUeE5wEQrPc=", 238 | "requires": { 239 | "inherits": "2.0.3", 240 | "readable-stream": "2.3.3", 241 | "typedarray": "0.0.6" 242 | }, 243 | "dependencies": { 244 | "isarray": { 245 | "version": "1.0.0", 246 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 247 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 248 | }, 249 | "readable-stream": { 250 | "version": "2.3.3", 251 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.3.tgz", 252 | "integrity": "sha512-m+qzzcn7KUxEmd1gMbchF+Y2eIUbieUaxkWtptyHywrX0rE8QEYqPC07Vuy4Wm32/xE16NcdBctb8S0Xe/5IeQ==", 253 | "requires": { 254 | "core-util-is": "1.0.2", 255 | "inherits": "2.0.3", 256 | "isarray": "1.0.0", 257 | "process-nextick-args": "1.0.7", 258 | "safe-buffer": "5.1.1", 259 | "string_decoder": "1.0.3", 260 | "util-deprecate": "1.0.2" 261 | } 262 | }, 263 | "string_decoder": { 264 | "version": "1.0.3", 265 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.0.3.tgz", 266 | "integrity": "sha512-4AH6Z5fzNNBcH+6XDMfA/BTt87skxqJlO0lAh3Dker5zThcAxG6mKz+iGu308UKoPPQ8Dcqx/4JhujzltRa+hQ==", 267 | "requires": { 268 | "safe-buffer": "5.1.1" 269 | } 270 | } 271 | } 272 | }, 273 | "config": { 274 | "version": "1.26.2", 275 | "resolved": "https://registry.npmjs.org/config/-/config-1.26.2.tgz", 276 | "integrity": "sha1-JGYpEWjYr64Kroq5nqTUJy9SDK4=", 277 | "requires": { 278 | "json5": "0.4.0", 279 | "os-homedir": "1.0.2" 280 | } 281 | }, 282 | "content-disposition": { 283 | "version": "0.5.2", 284 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.2.tgz", 285 | "integrity": "sha1-DPaLud318r55YcOoUXjLhdunjLQ=" 286 | }, 287 | "content-type": { 288 | "version": "1.0.4", 289 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", 290 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" 291 | }, 292 | "convert-to-ecmascript-compatible-varname": { 293 | "version": "0.1.5", 294 | "resolved": "https://registry.npmjs.org/convert-to-ecmascript-compatible-varname/-/convert-to-ecmascript-compatible-varname-0.1.5.tgz", 295 | "integrity": "sha1-9npJOMUjNENWQlBHnGcBS6yHhJk=" 296 | }, 297 | "cookie": { 298 | "version": "0.3.1", 299 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.3.1.tgz", 300 | "integrity": "sha1-5+Ch+e9DtMi6klxcWpboBtFoc7s=" 301 | }, 302 | "cookie-signature": { 303 | "version": "1.0.6", 304 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 305 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" 306 | }, 307 | "cookiejar": { 308 | "version": "2.1.1", 309 | "resolved": "https://registry.npmjs.org/cookiejar/-/cookiejar-2.1.1.tgz", 310 | "integrity": "sha1-Qa1XsbVVlR7BcUEqgZQrHoIA00o=" 311 | }, 312 | "core-util-is": { 313 | "version": "1.0.2", 314 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", 315 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" 316 | }, 317 | "cors": { 318 | "version": "2.8.4", 319 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.4.tgz", 320 | "integrity": "sha1-K9OB8usgECAQXNUOpZ2mMJBpRoY=", 321 | "requires": { 322 | "object-assign": "4.1.1", 323 | "vary": "1.1.1" 324 | } 325 | }, 326 | "cryptiles": { 327 | "version": "2.0.5", 328 | "resolved": "https://registry.npmjs.org/cryptiles/-/cryptiles-2.0.5.tgz", 329 | "integrity": "sha1-O9/s3GCBR8HGcgL6KR59ylnqo7g=", 330 | "requires": { 331 | "boom": "2.10.1" 332 | } 333 | }, 334 | "cycle": { 335 | "version": "1.0.3", 336 | "resolved": "https://registry.npmjs.org/cycle/-/cycle-1.0.3.tgz", 337 | "integrity": "sha1-IegLK+hYD5i0aPN5QwZisEbDStI=" 338 | }, 339 | "dashdash": { 340 | "version": "1.14.1", 341 | "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", 342 | "integrity": "sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=", 343 | "requires": { 344 | "assert-plus": "1.0.0" 345 | }, 346 | "dependencies": { 347 | "assert-plus": { 348 | "version": "1.0.0", 349 | "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", 350 | "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=" 351 | } 352 | } 353 | }, 354 | "debug": { 355 | "version": "2.6.8", 356 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.8.tgz", 357 | "integrity": "sha1-5zFTHKLt4n0YgiJCfaF4IdaP9Pw=", 358 | "requires": { 359 | "ms": "2.0.0" 360 | } 361 | }, 362 | "deep-extend": { 363 | "version": "0.4.2", 364 | "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.4.2.tgz", 365 | "integrity": "sha1-SLaZwn4zS/ifEIkr5DL25MfTSn8=" 366 | }, 367 | "delayed-stream": { 368 | "version": "1.0.0", 369 | "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", 370 | "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=" 371 | }, 372 | "depd": { 373 | "version": "1.1.1", 374 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.1.tgz", 375 | "integrity": "sha1-V4O04cRZ8G+lyif5kfPQbnoxA1k=" 376 | }, 377 | "deref": { 378 | "version": "0.6.4", 379 | "resolved": "https://registry.npmjs.org/deref/-/deref-0.6.4.tgz", 380 | "integrity": "sha1-vVqW1F2+0wEbuBvfaN31S+jhvU4=", 381 | "requires": { 382 | "deep-extend": "0.4.2" 383 | } 384 | }, 385 | "destroy": { 386 | "version": "1.0.4", 387 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", 388 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" 389 | }, 390 | "dicer": { 391 | "version": "0.2.5", 392 | "resolved": "https://registry.npmjs.org/dicer/-/dicer-0.2.5.tgz", 393 | "integrity": "sha1-WZbAhrszIYyBLAkL3cCc0S+stw8=", 394 | "requires": { 395 | "readable-stream": "1.1.14", 396 | "streamsearch": "0.1.2" 397 | } 398 | }, 399 | "discontinuous-range": { 400 | "version": "1.0.0", 401 | "resolved": "https://registry.npmjs.org/discontinuous-range/-/discontinuous-range-1.0.0.tgz", 402 | "integrity": "sha1-44Mx8IRLukm5qctxx3FYWqsbxlo=" 403 | }, 404 | "ecc-jsbn": { 405 | "version": "0.1.1", 406 | "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.1.tgz", 407 | "integrity": "sha1-D8c6ntXw1Tw4GTOYUj735UN3dQU=", 408 | "optional": true, 409 | "requires": { 410 | "jsbn": "0.1.1" 411 | } 412 | }, 413 | "ee-first": { 414 | "version": "1.1.1", 415 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 416 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" 417 | }, 418 | "encodeurl": { 419 | "version": "1.0.1", 420 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.1.tgz", 421 | "integrity": "sha1-eePVhlU0aQn+bw9Fpd5oEDspTSA=" 422 | }, 423 | "error": { 424 | "version": "7.0.2", 425 | "resolved": "https://registry.npmjs.org/error/-/error-7.0.2.tgz", 426 | "integrity": "sha1-pfdf/02ZJhJt2sDqXcOOaJFTywI=", 427 | "requires": { 428 | "string-template": "0.2.1", 429 | "xtend": "4.0.1" 430 | } 431 | }, 432 | "escape-html": { 433 | "version": "1.0.3", 434 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 435 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" 436 | }, 437 | "esprima": { 438 | "version": "4.0.0", 439 | "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.0.tgz", 440 | "integrity": "sha512-oftTcaMu/EGrEIu904mWteKIv8vMuOgGYo7EhVJJN00R/EED9DCua/xxHRdYnKtcECzVg7xOWhflvJMnqcFZjw==" 441 | }, 442 | "etag": { 443 | "version": "1.8.1", 444 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 445 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" 446 | }, 447 | "express": { 448 | "version": "4.15.4", 449 | "resolved": "https://registry.npmjs.org/express/-/express-4.15.4.tgz", 450 | "integrity": "sha1-Ay4iU0ic+PzgJma+yj0R7XotrtE=", 451 | "requires": { 452 | "accepts": "1.3.4", 453 | "array-flatten": "1.1.1", 454 | "content-disposition": "0.5.2", 455 | "content-type": "1.0.4", 456 | "cookie": "0.3.1", 457 | "cookie-signature": "1.0.6", 458 | "debug": "2.6.8", 459 | "depd": "1.1.1", 460 | "encodeurl": "1.0.1", 461 | "escape-html": "1.0.3", 462 | "etag": "1.8.1", 463 | "finalhandler": "1.0.5", 464 | "fresh": "0.5.0", 465 | "merge-descriptors": "1.0.1", 466 | "methods": "1.1.2", 467 | "on-finished": "2.3.0", 468 | "parseurl": "1.3.2", 469 | "path-to-regexp": "0.1.7", 470 | "proxy-addr": "1.1.5", 471 | "qs": "6.5.0", 472 | "range-parser": "1.2.0", 473 | "send": "0.15.4", 474 | "serve-static": "1.12.4", 475 | "setprototypeof": "1.0.3", 476 | "statuses": "1.3.1", 477 | "type-is": "1.6.15", 478 | "utils-merge": "1.0.0", 479 | "vary": "1.1.1" 480 | } 481 | }, 482 | "extend": { 483 | "version": "3.0.1", 484 | "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.1.tgz", 485 | "integrity": "sha1-p1Xqe8Gt/MWjHOfnYtuq3F5jZEQ=" 486 | }, 487 | "extsprintf": { 488 | "version": "1.3.0", 489 | "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", 490 | "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=" 491 | }, 492 | "eyes": { 493 | "version": "0.1.8", 494 | "resolved": "https://registry.npmjs.org/eyes/-/eyes-0.1.8.tgz", 495 | "integrity": "sha1-Ys8SAjTGg3hdkCNIqADvPgzCC8A=" 496 | }, 497 | "faker": { 498 | "version": "3.1.0", 499 | "resolved": "https://registry.npmjs.org/faker/-/faker-3.1.0.tgz", 500 | "integrity": "sha1-D5CPr05uwCUk5UpX5DLFwBPgjJ8=" 501 | }, 502 | "finalhandler": { 503 | "version": "1.0.5", 504 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.0.5.tgz", 505 | "integrity": "sha1-pwEwPSV6G8gv6lR6M+WuiVMXI98=", 506 | "requires": { 507 | "debug": "2.6.8", 508 | "encodeurl": "1.0.1", 509 | "escape-html": "1.0.3", 510 | "on-finished": "2.3.0", 511 | "parseurl": "1.3.2", 512 | "statuses": "1.3.1", 513 | "unpipe": "1.0.0" 514 | } 515 | }, 516 | "forever-agent": { 517 | "version": "0.6.1", 518 | "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", 519 | "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=" 520 | }, 521 | "form-data": { 522 | "version": "2.1.4", 523 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.1.4.tgz", 524 | "integrity": "sha1-M8GDrPGTJ27KqYFDpp6Uv+4XUNE=", 525 | "requires": { 526 | "asynckit": "0.4.0", 527 | "combined-stream": "1.0.5", 528 | "mime-types": "2.1.17" 529 | } 530 | }, 531 | "formidable": { 532 | "version": "1.1.1", 533 | "resolved": "https://registry.npmjs.org/formidable/-/formidable-1.1.1.tgz", 534 | "integrity": "sha1-lriIb3w8NQi5Mta9cMTTqI818ak=" 535 | }, 536 | "forwarded": { 537 | "version": "0.1.2", 538 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", 539 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=" 540 | }, 541 | "fresh": { 542 | "version": "0.5.0", 543 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.0.tgz", 544 | "integrity": "sha1-9HTKXmqSRtb9jglTz6m5yAWvp44=" 545 | }, 546 | "getpass": { 547 | "version": "0.1.7", 548 | "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", 549 | "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=", 550 | "requires": { 551 | "assert-plus": "1.0.0" 552 | }, 553 | "dependencies": { 554 | "assert-plus": { 555 | "version": "1.0.0", 556 | "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", 557 | "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=" 558 | } 559 | } 560 | }, 561 | "graphlib": { 562 | "version": "2.1.1", 563 | "resolved": "https://registry.npmjs.org/graphlib/-/graphlib-2.1.1.tgz", 564 | "integrity": "sha1-QjUsUrovTQNctWbrkfc5X3bryVE=", 565 | "requires": { 566 | "lodash": "4.17.4" 567 | }, 568 | "dependencies": { 569 | "lodash": { 570 | "version": "4.17.4", 571 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.4.tgz", 572 | "integrity": "sha1-eCA6TRwyiuHYbcpkYONptX9AVa4=" 573 | } 574 | } 575 | }, 576 | "har-schema": { 577 | "version": "1.0.5", 578 | "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-1.0.5.tgz", 579 | "integrity": "sha1-0mMTX0MwfALGAq/I/pWXDAFRNp4=" 580 | }, 581 | "har-validator": { 582 | "version": "4.2.1", 583 | "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-4.2.1.tgz", 584 | "integrity": "sha1-M0gdDxu/9gDdID11gSpqX7oALio=", 585 | "requires": { 586 | "ajv": "4.11.8", 587 | "har-schema": "1.0.5" 588 | } 589 | }, 590 | "hawk": { 591 | "version": "3.1.3", 592 | "resolved": "https://registry.npmjs.org/hawk/-/hawk-3.1.3.tgz", 593 | "integrity": "sha1-B4REvXwWQLD+VA0sm3PVlnjo4cQ=", 594 | "requires": { 595 | "boom": "2.10.1", 596 | "cryptiles": "2.0.5", 597 | "hoek": "2.16.3", 598 | "sntp": "1.0.9" 599 | } 600 | }, 601 | "hoek": { 602 | "version": "2.16.3", 603 | "resolved": "https://registry.npmjs.org/hoek/-/hoek-2.16.3.tgz", 604 | "integrity": "sha1-ILt0A9POo5jpHcRxCo/xuCdKJe0=" 605 | }, 606 | "http-errors": { 607 | "version": "1.6.2", 608 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.2.tgz", 609 | "integrity": "sha1-CgAsyFcHGSp+eUbO7cERVfYOxzY=", 610 | "requires": { 611 | "depd": "1.1.1", 612 | "inherits": "2.0.3", 613 | "setprototypeof": "1.0.3", 614 | "statuses": "1.3.1" 615 | } 616 | }, 617 | "http-signature": { 618 | "version": "1.1.1", 619 | "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.1.1.tgz", 620 | "integrity": "sha1-33LiZwZs0Kxn+3at+OE0qPvPkb8=", 621 | "requires": { 622 | "assert-plus": "0.2.0", 623 | "jsprim": "1.4.1", 624 | "sshpk": "1.13.1" 625 | } 626 | }, 627 | "iconv-lite": { 628 | "version": "0.4.19", 629 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.19.tgz", 630 | "integrity": "sha512-oTZqweIP51xaGPI4uPa56/Pri/480R+mo7SeU+YETByQNhDG55ycFyNLIgta9vXhILrxXDmF7ZGhqZIcuN0gJQ==" 631 | }, 632 | "inherits": { 633 | "version": "2.0.3", 634 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 635 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" 636 | }, 637 | "ipaddr.js": { 638 | "version": "1.4.0", 639 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.4.0.tgz", 640 | "integrity": "sha1-KWrKh4qCGBbluF0KKFqZvP9FgvA=" 641 | }, 642 | "is-typedarray": { 643 | "version": "1.0.0", 644 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", 645 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=" 646 | }, 647 | "isarray": { 648 | "version": "0.0.1", 649 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz", 650 | "integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8=" 651 | }, 652 | "isstream": { 653 | "version": "0.1.2", 654 | "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz", 655 | "integrity": "sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo=" 656 | }, 657 | "jaeger-client": { 658 | "version": "3.5.3", 659 | "resolved": "https://registry.npmjs.org/jaeger-client/-/jaeger-client-3.5.3.tgz", 660 | "integrity": "sha1-MKOYlkLuPo6gbRDCufngRtQk2O0=", 661 | "requires": { 662 | "node-int64": "0.4.0", 663 | "opentracing": "0.13.0", 664 | "thriftrw": "3.11.1", 665 | "xorshift": "0.2.1" 666 | }, 667 | "dependencies": { 668 | "opentracing": { 669 | "version": "0.13.0", 670 | "resolved": "https://registry.npmjs.org/opentracing/-/opentracing-0.13.0.tgz", 671 | "integrity": "sha1-ajQUQvCdfYZrwR7QPeHjgo49aqs=" 672 | } 673 | } 674 | }, 675 | "js-base64": { 676 | "version": "2.3.2", 677 | "resolved": "https://registry.npmjs.org/js-base64/-/js-base64-2.3.2.tgz", 678 | "integrity": "sha512-Y2/+DnfJJXT1/FCwUebUhLWb3QihxiSC42+ctHLGogmW2jPY6LCapMdFZXRvVP2z6qyKW7s6qncE/9gSqZiArw==" 679 | }, 680 | "js-yaml": { 681 | "version": "3.10.0", 682 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.10.0.tgz", 683 | "integrity": "sha512-O2v52ffjLa9VeM43J4XocZE//WT9N0IiwDa3KSHH7Tu8CtH+1qM8SIZvnsTh6v+4yFy5KUY3BHUVwjpfAWsjIA==", 684 | "requires": { 685 | "argparse": "1.0.9", 686 | "esprima": "4.0.0" 687 | } 688 | }, 689 | "jsbn": { 690 | "version": "0.1.1", 691 | "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", 692 | "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=", 693 | "optional": true 694 | }, 695 | "json-refs": { 696 | "version": "2.1.7", 697 | "resolved": "https://registry.npmjs.org/json-refs/-/json-refs-2.1.7.tgz", 698 | "integrity": "sha1-uesB/in16j6Sh48VrqEK04taz4k=", 699 | "requires": { 700 | "commander": "2.11.0", 701 | "graphlib": "2.1.1", 702 | "js-yaml": "3.10.0", 703 | "native-promise-only": "0.8.1", 704 | "path-loader": "1.0.2", 705 | "slash": "1.0.0", 706 | "uri-js": "3.0.2" 707 | } 708 | }, 709 | "json-schema": { 710 | "version": "0.2.3", 711 | "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.2.3.tgz", 712 | "integrity": "sha1-tIDIkuWaLwWVTOcnvT8qTogvnhM=" 713 | }, 714 | "json-schema-faker": { 715 | "version": "0.2.16", 716 | "resolved": "https://registry.npmjs.org/json-schema-faker/-/json-schema-faker-0.2.16.tgz", 717 | "integrity": "sha1-UdPKSJVdj+c09ZHXR7ckU75aePI=", 718 | "requires": { 719 | "chance": "1.0.11", 720 | "deref": "0.6.4", 721 | "faker": "3.1.0", 722 | "randexp": "0.4.6" 723 | } 724 | }, 725 | "json-stable-stringify": { 726 | "version": "1.0.1", 727 | "resolved": "https://registry.npmjs.org/json-stable-stringify/-/json-stable-stringify-1.0.1.tgz", 728 | "integrity": "sha1-mnWdOcXy/1A/1TAGRu1EX4jE+a8=", 729 | "requires": { 730 | "jsonify": "0.0.0" 731 | } 732 | }, 733 | "json-stringify-safe": { 734 | "version": "5.0.1", 735 | "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", 736 | "integrity": "sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=" 737 | }, 738 | "json5": { 739 | "version": "0.4.0", 740 | "resolved": "https://registry.npmjs.org/json5/-/json5-0.4.0.tgz", 741 | "integrity": "sha1-BUNS5MTIDIbAkjh31EneF2pzLI0=" 742 | }, 743 | "jsonify": { 744 | "version": "0.0.0", 745 | "resolved": "https://registry.npmjs.org/jsonify/-/jsonify-0.0.0.tgz", 746 | "integrity": "sha1-LHS27kHZPKUbe1qu6PUDYx0lKnM=" 747 | }, 748 | "jspath": { 749 | "version": "0.3.4", 750 | "resolved": "https://registry.npmjs.org/jspath/-/jspath-0.3.4.tgz", 751 | "integrity": "sha1-2J0+0uh0NP5s0ASyQskS35aXNSQ=" 752 | }, 753 | "jsprim": { 754 | "version": "1.4.1", 755 | "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.1.tgz", 756 | "integrity": "sha1-MT5mvB5cwG5Di8G3SZwuXFastqI=", 757 | "requires": { 758 | "assert-plus": "1.0.0", 759 | "extsprintf": "1.3.0", 760 | "json-schema": "0.2.3", 761 | "verror": "1.10.0" 762 | }, 763 | "dependencies": { 764 | "assert-plus": { 765 | "version": "1.0.0", 766 | "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", 767 | "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=" 768 | } 769 | } 770 | }, 771 | "lodash": { 772 | "version": "3.10.1", 773 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-3.10.1.tgz", 774 | "integrity": "sha1-W/Rejkm6QYnhfUgnid/RW9FAt7Y=" 775 | }, 776 | "lodash.get": { 777 | "version": "4.4.2", 778 | "resolved": "https://registry.npmjs.org/lodash.get/-/lodash.get-4.4.2.tgz", 779 | "integrity": "sha1-LRd/ZS+jHpObRDjVNBSZ36OCXpk=" 780 | }, 781 | "lodash.isequal": { 782 | "version": "4.5.0", 783 | "resolved": "https://registry.npmjs.org/lodash.isequal/-/lodash.isequal-4.5.0.tgz", 784 | "integrity": "sha1-QVxEePK8wwEgwizhDtMib30+GOA=" 785 | }, 786 | "long": { 787 | "version": "2.4.0", 788 | "resolved": "https://registry.npmjs.org/long/-/long-2.4.0.tgz", 789 | "integrity": "sha1-n6GAux2VAM3CnEFWdmoZleH0Uk8=" 790 | }, 791 | "machine": { 792 | "version": "10.4.0", 793 | "resolved": "https://registry.npmjs.org/machine/-/machine-10.4.0.tgz", 794 | "integrity": "sha1-m1Ys5GeCEzKCijd9GQ65NrTkB7I=", 795 | "requires": { 796 | "convert-to-ecmascript-compatible-varname": "0.1.5", 797 | "debug": "2.6.8", 798 | "lodash": "3.10.1", 799 | "object-hash": "0.3.0", 800 | "rttc": "7.4.0", 801 | "switchback": "2.0.2" 802 | } 803 | }, 804 | "machinepack-http": { 805 | "version": "2.4.0", 806 | "resolved": "https://registry.npmjs.org/machinepack-http/-/machinepack-http-2.4.0.tgz", 807 | "integrity": "sha1-CnhcF9xrnBuaxAiBvu+uiudIVek=", 808 | "requires": { 809 | "lodash": "3.10.1", 810 | "machine": "10.4.0", 811 | "machinepack-urls": "4.1.0", 812 | "request": "2.81.0" 813 | } 814 | }, 815 | "machinepack-urls": { 816 | "version": "4.1.0", 817 | "resolved": "https://registry.npmjs.org/machinepack-urls/-/machinepack-urls-4.1.0.tgz", 818 | "integrity": "sha1-0l4y6Xw8LLiVaLqMmNIp1cMF45E=", 819 | "requires": { 820 | "lodash": "3.10.1", 821 | "machine": "9.1.2" 822 | }, 823 | "dependencies": { 824 | "machine": { 825 | "version": "9.1.2", 826 | "resolved": "https://registry.npmjs.org/machine/-/machine-9.1.2.tgz", 827 | "integrity": "sha1-hL+Pt3ZqlqplqpbWbpUJ62oFqDQ=", 828 | "requires": { 829 | "convert-to-ecmascript-compatible-varname": "0.1.5", 830 | "debug": "2.6.8", 831 | "lodash": "3.10.1", 832 | "object-hash": "0.3.0", 833 | "rttc": "4.5.2", 834 | "switchback": "1.1.3" 835 | } 836 | }, 837 | "rttc": { 838 | "version": "4.5.2", 839 | "resolved": "https://registry.npmjs.org/rttc/-/rttc-4.5.2.tgz", 840 | "integrity": "sha1-umo+komLQnTxI7usSUhddhajfLw=", 841 | "requires": { 842 | "lodash": "3.10.1" 843 | } 844 | }, 845 | "switchback": { 846 | "version": "1.1.3", 847 | "resolved": "https://registry.npmjs.org/switchback/-/switchback-1.1.3.tgz", 848 | "integrity": "sha1-EscBCTSNailvc5upEO64U/i25jE=", 849 | "requires": { 850 | "lodash": "2.4.2" 851 | }, 852 | "dependencies": { 853 | "lodash": { 854 | "version": "2.4.2", 855 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-2.4.2.tgz", 856 | "integrity": "sha1-+t2DS5aDBz2hebPq5tnA0VBT9z4=" 857 | } 858 | } 859 | } 860 | } 861 | }, 862 | "media-typer": { 863 | "version": "0.3.0", 864 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 865 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" 866 | }, 867 | "merge-descriptors": { 868 | "version": "1.0.1", 869 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 870 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" 871 | }, 872 | "methods": { 873 | "version": "1.1.2", 874 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 875 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" 876 | }, 877 | "mime": { 878 | "version": "1.3.4", 879 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.3.4.tgz", 880 | "integrity": "sha1-EV+eO2s9rylZmDyzjxSaLUDrXVM=" 881 | }, 882 | "mime-db": { 883 | "version": "1.30.0", 884 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.30.0.tgz", 885 | "integrity": "sha1-dMZD2i3Z1qRTmZY0ZbJtXKfXHwE=" 886 | }, 887 | "mime-types": { 888 | "version": "2.1.17", 889 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.17.tgz", 890 | "integrity": "sha1-Cdejk/A+mVp5+K+Fe3Cp4KsWVXo=", 891 | "requires": { 892 | "mime-db": "1.30.0" 893 | } 894 | }, 895 | "minimist": { 896 | "version": "0.0.8", 897 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", 898 | "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=" 899 | }, 900 | "mkdirp": { 901 | "version": "0.5.1", 902 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", 903 | "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", 904 | "requires": { 905 | "minimist": "0.0.8" 906 | } 907 | }, 908 | "module-details-from-path": { 909 | "version": "1.0.3", 910 | "resolved": "https://registry.npmjs.org/module-details-from-path/-/module-details-from-path-1.0.3.tgz", 911 | "integrity": "sha1-EUyUlnPiqKNenTV4hSeqN7Z52is=" 912 | }, 913 | "ms": { 914 | "version": "2.0.0", 915 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 916 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 917 | }, 918 | "multer": { 919 | "version": "1.3.0", 920 | "resolved": "https://registry.npmjs.org/multer/-/multer-1.3.0.tgz", 921 | "integrity": "sha1-CSsmcPaEb6SRSWXvyM+Uwg/sbNI=", 922 | "requires": { 923 | "append-field": "0.1.0", 924 | "busboy": "0.2.14", 925 | "concat-stream": "1.6.0", 926 | "mkdirp": "0.5.1", 927 | "object-assign": "3.0.0", 928 | "on-finished": "2.3.0", 929 | "type-is": "1.6.15", 930 | "xtend": "4.0.1" 931 | }, 932 | "dependencies": { 933 | "object-assign": { 934 | "version": "3.0.0", 935 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-3.0.0.tgz", 936 | "integrity": "sha1-m+3VygiXlJvKR+f/QIBi1Un1h/I=" 937 | } 938 | } 939 | }, 940 | "mustache": { 941 | "version": "2.3.0", 942 | "resolved": "https://registry.npmjs.org/mustache/-/mustache-2.3.0.tgz", 943 | "integrity": "sha1-QCj3d4sXcIpImTCm5SrDvKDaQdA=" 944 | }, 945 | "native-promise-only": { 946 | "version": "0.8.1", 947 | "resolved": "https://registry.npmjs.org/native-promise-only/-/native-promise-only-0.8.1.tgz", 948 | "integrity": "sha1-IKMYwwy0X3H+et+/eyHJnBRy7xE=" 949 | }, 950 | "negotiator": { 951 | "version": "0.6.1", 952 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.1.tgz", 953 | "integrity": "sha1-KzJxhOiZIQEXeyhWP7XnECrNDKk=" 954 | }, 955 | "node-int64": { 956 | "version": "0.4.0", 957 | "resolved": "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz", 958 | "integrity": "sha1-h6kGXNs1XTGC2PlM4RGIuCXGijs=" 959 | }, 960 | "oauth-sign": { 961 | "version": "0.8.2", 962 | "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.8.2.tgz", 963 | "integrity": "sha1-Rqarfwrq2N6unsBWV4C31O/rnUM=" 964 | }, 965 | "object-assign": { 966 | "version": "4.1.1", 967 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 968 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=" 969 | }, 970 | "object-hash": { 971 | "version": "0.3.0", 972 | "resolved": "https://registry.npmjs.org/object-hash/-/object-hash-0.3.0.tgz", 973 | "integrity": "sha1-VIII5Ds2pE5NowutbFasU7iF50Q=" 974 | }, 975 | "on-finished": { 976 | "version": "2.3.0", 977 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", 978 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", 979 | "requires": { 980 | "ee-first": "1.1.1" 981 | } 982 | }, 983 | "opentracing": { 984 | "version": "0.14.1", 985 | "resolved": "https://registry.npmjs.org/opentracing/-/opentracing-0.14.1.tgz", 986 | "integrity": "sha1-QNJ4vupBdmCjXdnT7nZRH/qRHc0=" 987 | }, 988 | "os-homedir": { 989 | "version": "1.0.2", 990 | "resolved": "https://registry.npmjs.org/os-homedir/-/os-homedir-1.0.2.tgz", 991 | "integrity": "sha1-/7xJiDNuDoM94MFox+8VISGqf7M=" 992 | }, 993 | "parseurl": { 994 | "version": "1.3.2", 995 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.2.tgz", 996 | "integrity": "sha1-/CidTtiZMRlGDBViUyYs3I3mW/M=" 997 | }, 998 | "path-loader": { 999 | "version": "1.0.2", 1000 | "resolved": "https://registry.npmjs.org/path-loader/-/path-loader-1.0.2.tgz", 1001 | "integrity": "sha1-zVxz5+OakQEb4UjWv92KhbuTHvk=", 1002 | "requires": { 1003 | "native-promise-only": "0.8.1", 1004 | "superagent": "3.6.0" 1005 | } 1006 | }, 1007 | "path-parse": { 1008 | "version": "1.0.5", 1009 | "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.5.tgz", 1010 | "integrity": "sha1-PBrfhx6pzWyUMbbqK9dKD/BVxME=" 1011 | }, 1012 | "path-to-regexp": { 1013 | "version": "0.1.7", 1014 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 1015 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" 1016 | }, 1017 | "performance-now": { 1018 | "version": "0.2.0", 1019 | "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-0.2.0.tgz", 1020 | "integrity": "sha1-M+8wxcd9TqIcWlOGnZG1bY8lVeU=" 1021 | }, 1022 | "pipeworks": { 1023 | "version": "1.3.1", 1024 | "resolved": "https://registry.npmjs.org/pipeworks/-/pipeworks-1.3.1.tgz", 1025 | "integrity": "sha1-+ENvhWXtHZe/OoBjKlOXv9NTOF8=" 1026 | }, 1027 | "process-nextick-args": { 1028 | "version": "1.0.7", 1029 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-1.0.7.tgz", 1030 | "integrity": "sha1-FQ4gt1ZZCtP5EJPyWk8q2L/zC6M=" 1031 | }, 1032 | "proxy-addr": { 1033 | "version": "1.1.5", 1034 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-1.1.5.tgz", 1035 | "integrity": "sha1-ccDuOxAt4/IC87ZPYI0XP8uhqRg=", 1036 | "requires": { 1037 | "forwarded": "0.1.2", 1038 | "ipaddr.js": "1.4.0" 1039 | } 1040 | }, 1041 | "punycode": { 1042 | "version": "1.4.1", 1043 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz", 1044 | "integrity": "sha1-wNWmOycYgArY4esPpSachN1BhF4=" 1045 | }, 1046 | "qs": { 1047 | "version": "6.5.0", 1048 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.0.tgz", 1049 | "integrity": "sha512-fjVFjW9yhqMhVGwRExCXLhJKrLlkYSaxNWdyc9rmHlrVZbk35YHH312dFd7191uQeXkI3mKLZTIbSvIeFwFemg==" 1050 | }, 1051 | "randexp": { 1052 | "version": "0.4.6", 1053 | "resolved": "https://registry.npmjs.org/randexp/-/randexp-0.4.6.tgz", 1054 | "integrity": "sha512-80WNmd9DA0tmZrw9qQa62GPPWfuXJknrmVmLcxvq4uZBdYqb1wYoKTmnlGUchvVWe0XiLupYkBoXVOxz3C8DYQ==", 1055 | "requires": { 1056 | "discontinuous-range": "1.0.0", 1057 | "ret": "0.1.15" 1058 | } 1059 | }, 1060 | "range-parser": { 1061 | "version": "1.2.0", 1062 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.0.tgz", 1063 | "integrity": "sha1-9JvmtIeJTdxA3MlKMi9hEJLgDV4=" 1064 | }, 1065 | "raw-body": { 1066 | "version": "2.3.2", 1067 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.3.2.tgz", 1068 | "integrity": "sha1-vNYMd9Prk83gBQKVw/N5OJvIj4k=", 1069 | "requires": { 1070 | "bytes": "3.0.0", 1071 | "http-errors": "1.6.2", 1072 | "iconv-lite": "0.4.19", 1073 | "unpipe": "1.0.0" 1074 | } 1075 | }, 1076 | "readable-stream": { 1077 | "version": "1.1.14", 1078 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.1.14.tgz", 1079 | "integrity": "sha1-fPTFTvZI44EwhMY23SB54WbAgdk=", 1080 | "requires": { 1081 | "core-util-is": "1.0.2", 1082 | "inherits": "2.0.3", 1083 | "isarray": "0.0.1", 1084 | "string_decoder": "0.10.31" 1085 | } 1086 | }, 1087 | "request": { 1088 | "version": "2.81.0", 1089 | "resolved": "https://registry.npmjs.org/request/-/request-2.81.0.tgz", 1090 | "integrity": "sha1-xpKJRqDgbF+Nb4qTM0af/aRimKA=", 1091 | "requires": { 1092 | "aws-sign2": "0.6.0", 1093 | "aws4": "1.6.0", 1094 | "caseless": "0.12.0", 1095 | "combined-stream": "1.0.5", 1096 | "extend": "3.0.1", 1097 | "forever-agent": "0.6.1", 1098 | "form-data": "2.1.4", 1099 | "har-validator": "4.2.1", 1100 | "hawk": "3.1.3", 1101 | "http-signature": "1.1.1", 1102 | "is-typedarray": "1.0.0", 1103 | "isstream": "0.1.2", 1104 | "json-stringify-safe": "5.0.1", 1105 | "mime-types": "2.1.17", 1106 | "oauth-sign": "0.8.2", 1107 | "performance-now": "0.2.0", 1108 | "qs": "6.4.0", 1109 | "safe-buffer": "5.1.1", 1110 | "stringstream": "0.0.5", 1111 | "tough-cookie": "2.3.2", 1112 | "tunnel-agent": "0.6.0", 1113 | "uuid": "3.1.0" 1114 | }, 1115 | "dependencies": { 1116 | "qs": { 1117 | "version": "6.4.0", 1118 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.4.0.tgz", 1119 | "integrity": "sha1-E+JtKK1rD/qpExLNO/cI7TUecjM=" 1120 | } 1121 | } 1122 | }, 1123 | "require-in-the-middle": { 1124 | "version": "2.1.2", 1125 | "resolved": "https://registry.npmjs.org/require-in-the-middle/-/require-in-the-middle-2.1.2.tgz", 1126 | "integrity": "sha1-vduJMW1FvNsI4sYYa9Lm6Bmo7q4=", 1127 | "requires": { 1128 | "module-details-from-path": "1.0.3", 1129 | "resolve": "1.4.0" 1130 | } 1131 | }, 1132 | "resolve": { 1133 | "version": "1.4.0", 1134 | "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.4.0.tgz", 1135 | "integrity": "sha512-aW7sVKPufyHqOmyyLzg/J+8606v5nevBgaliIlV7nUpVMsDnoBGV/cbSLNjZAg9q0Cfd/+easKVKQ8vOu8fn1Q==", 1136 | "requires": { 1137 | "path-parse": "1.0.5" 1138 | } 1139 | }, 1140 | "ret": { 1141 | "version": "0.1.15", 1142 | "resolved": "https://registry.npmjs.org/ret/-/ret-0.1.15.tgz", 1143 | "integrity": "sha512-TTlYpa+OL+vMMNG24xSlQGEJ3B/RzEfUlLct7b5G/ytav+wPrplCpVMFuwzXbkecJrb6IYo1iFb0S9v37754mg==" 1144 | }, 1145 | "rttc": { 1146 | "version": "7.4.0", 1147 | "resolved": "https://registry.npmjs.org/rttc/-/rttc-7.4.0.tgz", 1148 | "integrity": "sha1-vJys1Grdkj3rYklaAZNOt+9hn7Q=", 1149 | "requires": { 1150 | "lodash": "3.10.1" 1151 | } 1152 | }, 1153 | "safe-buffer": { 1154 | "version": "5.1.1", 1155 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.1.tgz", 1156 | "integrity": "sha512-kKvNJn6Mm93gAczWVJg7wH+wGYWNrDHdWvpUmHyEsgCtIwwo3bqPtV4tR5tuPaUhTOo/kvhVwd8XwwOllGYkbg==" 1157 | }, 1158 | "semver": { 1159 | "version": "5.4.1", 1160 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.4.1.tgz", 1161 | "integrity": "sha512-WfG/X9+oATh81XtllIo/I8gOiY9EXRdv1cQdyykeXK17YcUW3EXUAi2To4pcH6nZtJPr7ZOpM5OMyWJZm+8Rsg==" 1162 | }, 1163 | "send": { 1164 | "version": "0.15.4", 1165 | "resolved": "https://registry.npmjs.org/send/-/send-0.15.4.tgz", 1166 | "integrity": "sha1-mF+qPihLAnPHkzZKNcZze9k5Bbk=", 1167 | "requires": { 1168 | "debug": "2.6.8", 1169 | "depd": "1.1.1", 1170 | "destroy": "1.0.4", 1171 | "encodeurl": "1.0.1", 1172 | "escape-html": "1.0.3", 1173 | "etag": "1.8.1", 1174 | "fresh": "0.5.0", 1175 | "http-errors": "1.6.2", 1176 | "mime": "1.3.4", 1177 | "ms": "2.0.0", 1178 | "on-finished": "2.3.0", 1179 | "range-parser": "1.2.0", 1180 | "statuses": "1.3.1" 1181 | } 1182 | }, 1183 | "serve-static": { 1184 | "version": "1.12.4", 1185 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.12.4.tgz", 1186 | "integrity": "sha1-m2qpjutyU8Tu3Ewfb9vKYJkBqWE=", 1187 | "requires": { 1188 | "encodeurl": "1.0.1", 1189 | "escape-html": "1.0.3", 1190 | "parseurl": "1.3.2", 1191 | "send": "0.15.4" 1192 | } 1193 | }, 1194 | "setprototypeof": { 1195 | "version": "1.0.3", 1196 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.0.3.tgz", 1197 | "integrity": "sha1-ZlZ+NwQ+608E2RvWWMDL77VbjgQ=" 1198 | }, 1199 | "shimmer": { 1200 | "version": "1.1.0", 1201 | "resolved": "https://registry.npmjs.org/shimmer/-/shimmer-1.1.0.tgz", 1202 | "integrity": "sha1-l9c3cTf/u6tCVSLkKf4KqJpIizU=" 1203 | }, 1204 | "slash": { 1205 | "version": "1.0.0", 1206 | "resolved": "https://registry.npmjs.org/slash/-/slash-1.0.0.tgz", 1207 | "integrity": "sha1-xB8vbDn8FtHNF61LXYlhFK5HDVU=" 1208 | }, 1209 | "sntp": { 1210 | "version": "1.0.9", 1211 | "resolved": "https://registry.npmjs.org/sntp/-/sntp-1.0.9.tgz", 1212 | "integrity": "sha1-ZUEYTMkK7qbG57NeJlkIJEPGYZg=", 1213 | "requires": { 1214 | "hoek": "2.16.3" 1215 | } 1216 | }, 1217 | "sprintf-js": { 1218 | "version": "1.0.3", 1219 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", 1220 | "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=" 1221 | }, 1222 | "sshpk": { 1223 | "version": "1.13.1", 1224 | "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.13.1.tgz", 1225 | "integrity": "sha1-US322mKHFEMW3EwY/hzx2UBzm+M=", 1226 | "requires": { 1227 | "asn1": "0.2.3", 1228 | "assert-plus": "1.0.0", 1229 | "bcrypt-pbkdf": "1.0.1", 1230 | "dashdash": "1.14.1", 1231 | "ecc-jsbn": "0.1.1", 1232 | "getpass": "0.1.7", 1233 | "jsbn": "0.1.1", 1234 | "tweetnacl": "0.14.5" 1235 | }, 1236 | "dependencies": { 1237 | "assert-plus": { 1238 | "version": "1.0.0", 1239 | "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", 1240 | "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=" 1241 | } 1242 | } 1243 | }, 1244 | "stack-chain": { 1245 | "version": "1.3.7", 1246 | "resolved": "https://registry.npmjs.org/stack-chain/-/stack-chain-1.3.7.tgz", 1247 | "integrity": "sha1-0ZLJ/06moiyUxN1FkXHj8AzqEoU=" 1248 | }, 1249 | "stack-trace": { 1250 | "version": "0.0.10", 1251 | "resolved": "https://registry.npmjs.org/stack-trace/-/stack-trace-0.0.10.tgz", 1252 | "integrity": "sha1-VHxws0fo0ytOEI6hoqFZ5f3eGcA=" 1253 | }, 1254 | "statuses": { 1255 | "version": "1.3.1", 1256 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.3.1.tgz", 1257 | "integrity": "sha1-+vUbnrdKrvOzrPStX2Gr8ky3uT4=" 1258 | }, 1259 | "streamsearch": { 1260 | "version": "0.1.2", 1261 | "resolved": "https://registry.npmjs.org/streamsearch/-/streamsearch-0.1.2.tgz", 1262 | "integrity": "sha1-gIudDlb8Jz2Am6VzOOkpkZoanxo=" 1263 | }, 1264 | "string_decoder": { 1265 | "version": "0.10.31", 1266 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-0.10.31.tgz", 1267 | "integrity": "sha1-YuIDvEF2bGwoyfyEMB2rHFMQ+pQ=" 1268 | }, 1269 | "string-template": { 1270 | "version": "0.2.1", 1271 | "resolved": "https://registry.npmjs.org/string-template/-/string-template-0.2.1.tgz", 1272 | "integrity": "sha1-QpMuWYo1LQH8IuwzZ9nYTuxsmt0=" 1273 | }, 1274 | "stringstream": { 1275 | "version": "0.0.5", 1276 | "resolved": "https://registry.npmjs.org/stringstream/-/stringstream-0.0.5.tgz", 1277 | "integrity": "sha1-TkhM1N5aC7vuGORjB3EKioFiGHg=" 1278 | }, 1279 | "superagent": { 1280 | "version": "3.6.0", 1281 | "resolved": "https://registry.npmjs.org/superagent/-/superagent-3.6.0.tgz", 1282 | "integrity": "sha512-oWsu4mboo8sVxagp4bNwZIR1rUmypeAJDmNIwT9mF4k06hSu6P92aOjEWLaIj7vsX3fOUp+cRH/04tao+q5Q7A==", 1283 | "requires": { 1284 | "component-emitter": "1.2.1", 1285 | "cookiejar": "2.1.1", 1286 | "debug": "2.6.8", 1287 | "extend": "3.0.1", 1288 | "form-data": "2.1.4", 1289 | "formidable": "1.1.1", 1290 | "methods": "1.1.2", 1291 | "mime": "1.4.0", 1292 | "qs": "6.5.0", 1293 | "readable-stream": "2.3.3" 1294 | }, 1295 | "dependencies": { 1296 | "isarray": { 1297 | "version": "1.0.0", 1298 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 1299 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 1300 | }, 1301 | "mime": { 1302 | "version": "1.4.0", 1303 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.4.0.tgz", 1304 | "integrity": "sha512-n9ChLv77+QQEapYz8lV+rIZAW3HhAPW2CXnzb1GN5uMkuczshwvkW7XPsbzU0ZQN3sP47Er2KVkp2p3KyqZKSQ==" 1305 | }, 1306 | "readable-stream": { 1307 | "version": "2.3.3", 1308 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.3.tgz", 1309 | "integrity": "sha512-m+qzzcn7KUxEmd1gMbchF+Y2eIUbieUaxkWtptyHywrX0rE8QEYqPC07Vuy4Wm32/xE16NcdBctb8S0Xe/5IeQ==", 1310 | "requires": { 1311 | "core-util-is": "1.0.2", 1312 | "inherits": "2.0.3", 1313 | "isarray": "1.0.0", 1314 | "process-nextick-args": "1.0.7", 1315 | "safe-buffer": "5.1.1", 1316 | "string_decoder": "1.0.3", 1317 | "util-deprecate": "1.0.2" 1318 | } 1319 | }, 1320 | "string_decoder": { 1321 | "version": "1.0.3", 1322 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.0.3.tgz", 1323 | "integrity": "sha512-4AH6Z5fzNNBcH+6XDMfA/BTt87skxqJlO0lAh3Dker5zThcAxG6mKz+iGu308UKoPPQ8Dcqx/4JhujzltRa+hQ==", 1324 | "requires": { 1325 | "safe-buffer": "5.1.1" 1326 | } 1327 | } 1328 | } 1329 | }, 1330 | "swagger-express-mw": { 1331 | "version": "0.7.0", 1332 | "resolved": "https://registry.npmjs.org/swagger-express-mw/-/swagger-express-mw-0.7.0.tgz", 1333 | "integrity": "sha1-SfXbctHUs4JzNu5sx7Npyu1b9Mg=", 1334 | "requires": { 1335 | "swagger-node-runner": "0.7.3" 1336 | } 1337 | }, 1338 | "swagger-methods": { 1339 | "version": "1.0.0", 1340 | "resolved": "https://registry.npmjs.org/swagger-methods/-/swagger-methods-1.0.0.tgz", 1341 | "integrity": "sha1-s5x3lX0wWmU1wKHgFQgRhbmdYfw=" 1342 | }, 1343 | "swagger-node-runner": { 1344 | "version": "0.7.3", 1345 | "resolved": "https://registry.npmjs.org/swagger-node-runner/-/swagger-node-runner-0.7.3.tgz", 1346 | "integrity": "sha1-P0RH+ma8Mv9Kmm+qw8rVVnst3Go=", 1347 | "requires": { 1348 | "async": "1.5.2", 1349 | "bagpipes": "0.1.0", 1350 | "body-parser": "1.18.1", 1351 | "config": "1.26.2", 1352 | "cors": "2.8.4", 1353 | "debug": "2.6.8", 1354 | "js-yaml": "3.10.0", 1355 | "lodash": "3.10.1", 1356 | "multer": "1.3.0", 1357 | "parseurl": "1.3.2", 1358 | "qs": "6.5.0", 1359 | "sway": "1.0.0", 1360 | "type-is": "1.6.15" 1361 | } 1362 | }, 1363 | "swagger-schema-official": { 1364 | "version": "2.0.0-bab6bed", 1365 | "resolved": "https://registry.npmjs.org/swagger-schema-official/-/swagger-schema-official-2.0.0-bab6bed.tgz", 1366 | "integrity": "sha1-cAcEaNbSl3ylI3suUZyn0Gouo/0=" 1367 | }, 1368 | "swagger-ui-dist": { 1369 | "version": "3.2.1", 1370 | "resolved": "https://registry.npmjs.org/swagger-ui-dist/-/swagger-ui-dist-3.2.1.tgz", 1371 | "integrity": "sha1-vbrheq4BcpgZMAalh9cje1iodc4=" 1372 | }, 1373 | "sway": { 1374 | "version": "1.0.0", 1375 | "resolved": "https://registry.npmjs.org/sway/-/sway-1.0.0.tgz", 1376 | "integrity": "sha1-No/8Dpa9hCJu0bmzPWa+V9oE8Jo=", 1377 | "requires": { 1378 | "debug": "2.6.8", 1379 | "js-base64": "2.3.2", 1380 | "js-yaml": "3.10.0", 1381 | "json-refs": "2.1.7", 1382 | "json-schema-faker": "0.2.16", 1383 | "lodash": "4.17.4", 1384 | "native-promise-only": "0.8.1", 1385 | "path-to-regexp": "1.7.0", 1386 | "swagger-methods": "1.0.0", 1387 | "swagger-schema-official": "2.0.0-bab6bed", 1388 | "z-schema": "3.18.4" 1389 | }, 1390 | "dependencies": { 1391 | "lodash": { 1392 | "version": "4.17.4", 1393 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.4.tgz", 1394 | "integrity": "sha1-eCA6TRwyiuHYbcpkYONptX9AVa4=" 1395 | }, 1396 | "path-to-regexp": { 1397 | "version": "1.7.0", 1398 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-1.7.0.tgz", 1399 | "integrity": "sha1-Wf3g9DW62suhA6hOnTvGTpa5k30=", 1400 | "requires": { 1401 | "isarray": "0.0.1" 1402 | } 1403 | } 1404 | } 1405 | }, 1406 | "switchback": { 1407 | "version": "2.0.2", 1408 | "resolved": "https://registry.npmjs.org/switchback/-/switchback-2.0.2.tgz", 1409 | "integrity": "sha1-ls8ODTY7VZ0Lt/8htip6qRDsYHk=", 1410 | "requires": { 1411 | "lodash": "3.10.1" 1412 | } 1413 | }, 1414 | "thriftrw": { 1415 | "version": "3.11.1", 1416 | "resolved": "https://registry.npmjs.org/thriftrw/-/thriftrw-3.11.1.tgz", 1417 | "integrity": "sha1-Wi9RZdZluxleZl5bW5+Il9rCOsw=", 1418 | "requires": { 1419 | "bufrw": "1.2.1", 1420 | "error": "7.0.2", 1421 | "long": "2.4.0" 1422 | } 1423 | }, 1424 | "tough-cookie": { 1425 | "version": "2.3.2", 1426 | "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.3.2.tgz", 1427 | "integrity": "sha1-8IH3bkyFcg5sN6X6ztc3FQ2EByo=", 1428 | "requires": { 1429 | "punycode": "1.4.1" 1430 | } 1431 | }, 1432 | "tunnel-agent": { 1433 | "version": "0.6.0", 1434 | "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", 1435 | "integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=", 1436 | "requires": { 1437 | "safe-buffer": "5.1.1" 1438 | } 1439 | }, 1440 | "tweetnacl": { 1441 | "version": "0.14.5", 1442 | "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz", 1443 | "integrity": "sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q=", 1444 | "optional": true 1445 | }, 1446 | "type-is": { 1447 | "version": "1.6.15", 1448 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.15.tgz", 1449 | "integrity": "sha1-yrEPtJCeRByChC6v4a1kbIGARBA=", 1450 | "requires": { 1451 | "media-typer": "0.3.0", 1452 | "mime-types": "2.1.17" 1453 | } 1454 | }, 1455 | "typedarray": { 1456 | "version": "0.0.6", 1457 | "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", 1458 | "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=" 1459 | }, 1460 | "unpipe": { 1461 | "version": "1.0.0", 1462 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 1463 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" 1464 | }, 1465 | "uri-js": { 1466 | "version": "3.0.2", 1467 | "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-3.0.2.tgz", 1468 | "integrity": "sha1-+QuFhQf4HepNz7s8TD2/orVX+qo=", 1469 | "requires": { 1470 | "punycode": "2.1.0" 1471 | }, 1472 | "dependencies": { 1473 | "punycode": { 1474 | "version": "2.1.0", 1475 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.0.tgz", 1476 | "integrity": "sha1-X4Y+3Im5bbCQdLrXlHvwkFbKTn0=" 1477 | } 1478 | } 1479 | }, 1480 | "util-deprecate": { 1481 | "version": "1.0.2", 1482 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", 1483 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" 1484 | }, 1485 | "utils-merge": { 1486 | "version": "1.0.0", 1487 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.0.tgz", 1488 | "integrity": "sha1-ApT7kiu5N1FTVBxPcJYjHyh8ivg=" 1489 | }, 1490 | "uuid": { 1491 | "version": "3.1.0", 1492 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.1.0.tgz", 1493 | "integrity": "sha512-DIWtzUkw04M4k3bf1IcpS2tngXEL26YUD2M0tMDUpnUrz2hgzUBlD55a4FjdLGPvfHxS6uluGWvaVEqgBcVa+g==" 1494 | }, 1495 | "validator": { 1496 | "version": "8.2.0", 1497 | "resolved": "https://registry.npmjs.org/validator/-/validator-8.2.0.tgz", 1498 | "integrity": "sha512-Yw5wW34fSv5spzTXNkokD6S6/Oq92d8q/t14TqsS3fAiA1RYnxSFSIZ+CY3n6PGGRCq5HhJTSepQvFUS2QUDxA==" 1499 | }, 1500 | "vary": { 1501 | "version": "1.1.1", 1502 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.1.tgz", 1503 | "integrity": "sha1-Z1Neu2lMHVIldFeYRmUyP1h+jTc=" 1504 | }, 1505 | "verror": { 1506 | "version": "1.10.0", 1507 | "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz", 1508 | "integrity": "sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA=", 1509 | "requires": { 1510 | "assert-plus": "1.0.0", 1511 | "core-util-is": "1.0.2", 1512 | "extsprintf": "1.3.0" 1513 | }, 1514 | "dependencies": { 1515 | "assert-plus": { 1516 | "version": "1.0.0", 1517 | "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", 1518 | "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=" 1519 | } 1520 | } 1521 | }, 1522 | "winston": { 1523 | "version": "2.3.1", 1524 | "resolved": "https://registry.npmjs.org/winston/-/winston-2.3.1.tgz", 1525 | "integrity": "sha1-C0hCDZeMAYBM8CMLZIhhWYIloRk=", 1526 | "requires": { 1527 | "async": "1.0.0", 1528 | "colors": "1.0.3", 1529 | "cycle": "1.0.3", 1530 | "eyes": "0.1.8", 1531 | "isstream": "0.1.2", 1532 | "stack-trace": "0.0.10" 1533 | }, 1534 | "dependencies": { 1535 | "async": { 1536 | "version": "1.0.0", 1537 | "resolved": "https://registry.npmjs.org/async/-/async-1.0.0.tgz", 1538 | "integrity": "sha1-+PwEyjoTeErenhZBr5hXjPvWR6k=" 1539 | } 1540 | } 1541 | }, 1542 | "xorshift": { 1543 | "version": "0.2.1", 1544 | "resolved": "https://registry.npmjs.org/xorshift/-/xorshift-0.2.1.tgz", 1545 | "integrity": "sha1-/NgiZ+k1HBPw+5xzMH8lMx0pxjo=" 1546 | }, 1547 | "xtend": { 1548 | "version": "4.0.1", 1549 | "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.1.tgz", 1550 | "integrity": "sha1-pcbVMr5lbiPbgg77lDofBJmNY68=" 1551 | }, 1552 | "z-schema": { 1553 | "version": "3.18.4", 1554 | "resolved": "https://registry.npmjs.org/z-schema/-/z-schema-3.18.4.tgz", 1555 | "integrity": "sha512-DUOKC/IhbkdLKKiV89gw9DUauTV8U/8yJl1sjf6MtDmzevLKOF2duNJ495S3MFVjqZarr+qNGCPbkg4mu4PpLw==", 1556 | "requires": { 1557 | "commander": "2.11.0", 1558 | "lodash.get": "4.4.2", 1559 | "lodash.isequal": "4.5.0", 1560 | "validator": "8.2.0" 1561 | } 1562 | } 1563 | } 1564 | } 1565 | -------------------------------------------------------------------------------- /services/vehicle-api/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "user-api", 3 | "version": "0.0.1", 4 | "private": true, 5 | "description": "User API", 6 | "main": "index.js", 7 | "scripts": { 8 | "start": "node index.js" 9 | }, 10 | "dependencies": { 11 | "@risingstack/opentracing-auto": "1.4.3", 12 | "express": "4.15.4", 13 | "jaeger-client": "3.5.3", 14 | "swagger-express-mw": "0.7.0", 15 | "swagger-ui-dist": "3.2.1", 16 | "winston": "2.3.1" 17 | } 18 | } 19 | -------------------------------------------------------------------------------- /services/vehicle-api/server.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | const logger = require('winston') 4 | const app = require('./app') 5 | 6 | const port = process.env.PORT || 3000 7 | const logLevel = process.env.LOG_LEVEL || 'debug' 8 | 9 | logger.level = logLevel 10 | 11 | const server = app.listen(port, (err) => { 12 | if (err) { 13 | logger.error(err) 14 | return 15 | } 16 | 17 | logger.info(`Server is listening on ${port}`) 18 | }) 19 | 20 | process.on('SIGTERM', () => { 21 | logger.info('SIGTERM received') 22 | 23 | server.close((err) => { 24 | if (err) { 25 | logger.error('Graceful shutdown', err) 26 | process.exit(1) 27 | } 28 | 29 | logger.info('Server stopped') 30 | process.exit(0) 31 | }) 32 | }) 33 | --------------------------------------------------------------------------------