├── .gitignore
├── lerna.json
├── packages
├── job-service
│ ├── src
│ │ ├── utils
│ │ │ ├── jobGenerator
│ │ │ │ ├── index.js
│ │ │ │ ├── jobId.js
│ │ │ │ ├── generateSalaryRange.js
│ │ │ │ ├── generateJob.js
│ │ │ │ └── generateTitle.js
│ │ │ └── getQueryParams.js
│ │ ├── index.js
│ │ ├── models
│ │ │ └── JobModel.js
│ │ └── controllers
│ │ │ └── Job.js
│ ├── package.json
│ └── package-lock.json
├── web-ui
│ ├── .babelrc
│ ├── enzyme.setup.js
│ ├── src
│ │ ├── __mocks__
│ │ │ └── react-router-dom.js
│ │ ├── components
│ │ │ ├── Loading.jsx
│ │ │ ├── NotFound.jsx
│ │ │ └── Salary.jsx
│ │ ├── index.html
│ │ ├── services
│ │ │ ├── fetchJobList.js
│ │ │ ├── fetchJob.js
│ │ │ └── __tests__
│ │ │ │ └── services.test.js
│ │ ├── pages
│ │ │ ├── __tests__
│ │ │ │ ├── __snapshots__
│ │ │ │ │ ├── JobPage.test.jsx.snap
│ │ │ │ │ └── JobListPage.test.jsx.snap
│ │ │ │ ├── JobPage.test.jsx
│ │ │ │ └── JobListPage.test.jsx
│ │ │ ├── JobListPage.jsx
│ │ │ └── JobPage.jsx
│ │ ├── index.jsx
│ │ └── style.scss
│ ├── pact.setup.js
│ ├── package.json
│ └── webpack.config.js
└── backend-for-webui
│ ├── src
│ ├── utils
│ │ └── getQueryParams.js
│ ├── index.js
│ └── controllers
│ │ ├── Job.js
│ │ └── __tests__
│ │ └── Job.test.js
│ ├── pact.setup.js
│ └── package.json
├── .editorconfig
├── package.json
├── README.md
├── .travis.yml
├── LICENSE
└── pacts
├── web-ui-backend-for-webui.json
└── backend-for-webui-job-service.json
/.gitignore:
--------------------------------------------------------------------------------
1 | node_modules
2 | target
3 | logs
4 | *.log
--------------------------------------------------------------------------------
/lerna.json:
--------------------------------------------------------------------------------
1 | {
2 | "lerna": "2.7.0",
3 | "packages": [
4 | "packages/*"
5 | ],
6 | "version": "0.0.0"
7 | }
8 |
--------------------------------------------------------------------------------
/packages/job-service/src/utils/jobGenerator/index.js:
--------------------------------------------------------------------------------
1 | const generateJob = require('./generateJob');
2 | module.exports = generateJob;
--------------------------------------------------------------------------------
/packages/web-ui/.babelrc:
--------------------------------------------------------------------------------
1 | {
2 | "presets": [
3 | "@babel/preset-react",
4 | "@babel/preset-env"
5 | ]
6 | }
--------------------------------------------------------------------------------
/.editorconfig:
--------------------------------------------------------------------------------
1 | root = true
2 |
3 | [*]
4 | end_of_line = lf
5 | insert_final_newline = true
6 | charset = utf-8
7 | indent_style = space
8 | indent_size = 4
9 |
--------------------------------------------------------------------------------
/packages/web-ui/enzyme.setup.js:
--------------------------------------------------------------------------------
1 | import Enzyme from 'enzyme';
2 | import Adapter from 'enzyme-adapter-react-16';
3 |
4 | Enzyme.configure({ adapter: new Adapter() });
--------------------------------------------------------------------------------
/packages/web-ui/src/__mocks__/react-router-dom.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 |
3 | const Link = (props) => ({props.children});
4 |
5 | module.exports = {
6 | Link
7 | }
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "contract-tests-example",
3 | "scripts": {
4 | "test": "lerna run test --stream --concurrency 1"
5 | },
6 | "devDependencies": {
7 | "lerna": "^2.7.0"
8 | }
9 | }
10 |
--------------------------------------------------------------------------------
/packages/web-ui/src/components/Loading.jsx:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import ReactDOM from 'react-dom';
3 |
4 | const Loading = () => {
5 | return Loading…
6 | }
7 |
8 | export default Loading;
--------------------------------------------------------------------------------
/packages/web-ui/src/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | Web Ui
6 |
7 |
8 |
9 |
10 |
--------------------------------------------------------------------------------
/packages/web-ui/src/services/fetchJobList.js:
--------------------------------------------------------------------------------
1 | import axios from 'axios';
2 |
3 | const serviceUrl = 'http://localhost:8080';
4 |
5 | const fetchJobList = () => {
6 | const url = `${serviceUrl}/v1/jobs`;
7 | return axios.get(url)
8 | .then(response => response.data);
9 | }
10 |
11 | export default fetchJobList;
--------------------------------------------------------------------------------
/packages/job-service/src/utils/getQueryParams.js:
--------------------------------------------------------------------------------
1 | const { parse } = require('querystring');
2 |
3 | function getQueryParams(url) {
4 | const position = url.indexOf('?');
5 |
6 | if (position < 0) {
7 | return {};
8 | }
9 |
10 | return parse(url.substr(position + 1));
11 | }
12 |
13 | module.exports = getQueryParams;
--------------------------------------------------------------------------------
/packages/web-ui/src/services/fetchJob.js:
--------------------------------------------------------------------------------
1 | import axios from 'axios';
2 |
3 | const serviceUrl = 'http://localhost:8080';
4 |
5 | const fetchJob = ({ id, slug }) => {
6 | const url = `${serviceUrl}/v1/job/${id}-${slug}`;
7 | return axios.get(url)
8 | .then(response => response.data);
9 | }
10 |
11 | export default fetchJob;
--------------------------------------------------------------------------------
/packages/backend-for-webui/src/utils/getQueryParams.js:
--------------------------------------------------------------------------------
1 | const { parse } = require('querystring');
2 |
3 | function getQueryParams(url) {
4 | const position = url.indexOf('?');
5 |
6 | if (position < 0) {
7 | return {};
8 | }
9 |
10 | return parse(url.substr(position + 1));
11 | }
12 |
13 | module.exports = getQueryParams;
--------------------------------------------------------------------------------
/packages/web-ui/src/components/NotFound.jsx:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import ReactDOM from 'react-dom';
3 | import { Link } from 'react-router-dom';
4 |
5 | const NotFound = () => {
6 | return (
7 |
8 |
Not Found
9 | Proceed to job listing
10 |
11 | )
12 | }
13 |
14 | export default NotFound;
--------------------------------------------------------------------------------
/packages/job-service/src/utils/jobGenerator/jobId.js:
--------------------------------------------------------------------------------
1 | let instance = null;
2 |
3 | class JobId {
4 | constructor() {
5 | this.id = 23450;
6 | }
7 |
8 | nextId() {
9 | return ++this.id;
10 | }
11 | }
12 |
13 | function getInstance() {
14 | if (! instance) {
15 | instance = new JobId();
16 | }
17 | return instance;
18 | }
19 |
20 | module.exports = getInstance;
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # contract-tests-example [](https://travis-ci.org/mateuszsokola/contract-tests-example)
2 |
3 |
4 | # How to run?
5 | ```
6 | npm i
7 | npm test
8 | ```
9 |
10 | # How to run tests for Job Service ?
11 | ```
12 | cd packages/job-service
13 | node src/index.js & # server must be running to verify provider
14 | npm run test:provider
15 | ```
16 |
--------------------------------------------------------------------------------
/.travis.yml:
--------------------------------------------------------------------------------
1 | sudo: false
2 | language: node_js
3 | node_js:
4 | - "node"
5 | cache:
6 | directories:
7 | - packages/backend-for-webui/node_modules
8 | - packages/job-service/node_modules
9 | - packages/web-ui/node_modules
10 | before_script:
11 | - lerna bootstrap
12 | script:
13 | - npm test
14 | - cd packages/job-service
15 | - node src/index.js &
16 | - npm run test:provider
17 | branches:
18 | only:
19 | - master
20 | - wip
--------------------------------------------------------------------------------
/packages/backend-for-webui/src/index.js:
--------------------------------------------------------------------------------
1 | const express = require('express');
2 | const cors = require('cors');
3 | const path = require('path');
4 |
5 | const { retrieveList, retrieveJob } = require('./controllers/Job');
6 |
7 | const port = 8080;
8 | const app = express();
9 |
10 | app.use(cors());
11 |
12 | app.get('/v1/jobs', retrieveList);
13 | app.get('/v1/job/:id-:slug', retrieveJob);
14 |
15 | app.listen(port, () => {
16 | console.log('BFF: Now listening on port', port);
17 | });
--------------------------------------------------------------------------------
/packages/web-ui/pact.setup.js:
--------------------------------------------------------------------------------
1 | const { Pact } = require('@pact-foundation/pact');
2 | const path = require('path');
3 |
4 | global.port = 8080;
5 | global.provider = new Pact({
6 | port: global.port,
7 | log: path.resolve(__dirname, '..', '..', 'logs', 'mockserver-integration.log'),
8 | dir: path.resolve(__dirname, '..', '..', 'pacts'),
9 | spec: 2,
10 | cors: true,
11 | pactfileWriteMode: 'update',
12 | consumer: 'web-ui',
13 | provider: 'backend-for-webui',
14 | logLevel: 'warn'
15 | });
16 |
--------------------------------------------------------------------------------
/packages/job-service/src/utils/jobGenerator/generateSalaryRange.js:
--------------------------------------------------------------------------------
1 | const { random } = require('lodash');
2 |
3 | const MIN_ANNUAL_SALARY = 3500000; // in cents
4 | const MAX_ANNUAL_SALARY = 8400000; // in cents
5 | const MULTIPLIER = 140000; // in cents
6 |
7 | function generateSalaryRange() {
8 | return {
9 | from: MIN_ANNUAL_SALARY + MULTIPLIER * random(0, 10),
10 | to: MAX_ANNUAL_SALARY - MULTIPLIER * random(0, 24),
11 | }
12 | }
13 |
14 | module.exports = generateSalaryRange;
--------------------------------------------------------------------------------
/packages/backend-for-webui/pact.setup.js:
--------------------------------------------------------------------------------
1 | const { Pact } = require('@pact-foundation/pact');
2 | const path = require('path');
3 |
4 | global.port = 3001;
5 | global.provider = new Pact({
6 | port: global.port,
7 | log: path.resolve(__dirname, '..', '..', 'logs', 'mockserver-integration.log'),
8 | dir: path.resolve(__dirname, '..', '..', 'pacts'),
9 | spec: 2,
10 | cors: true,
11 | pactfileWriteMode: 'update',
12 | consumer: 'backend-for-webui',
13 | provider: 'job-service',
14 | logLevel: 'warn'
15 | });
16 |
--------------------------------------------------------------------------------
/packages/job-service/src/index.js:
--------------------------------------------------------------------------------
1 | const express = require('express');
2 | const path = require('path');
3 |
4 | const { retrieveJobList, retrieveJob, setupTestData } = require('./controllers/Job');
5 |
6 | const port = 3001;
7 | const app = express();
8 |
9 | app.get('/v1', retrieveJobList);
10 | app.get('/v1/:id-:slug', retrieveJob);
11 |
12 | if (process.env.NODE_ENV !== 'production') {
13 | app.post('/setup', setupTestData);
14 | }
15 |
16 | app.listen(port, () => {
17 | console.log('Job Service: Now listening on port', port);
18 | });
19 |
20 | module.exports = app;
--------------------------------------------------------------------------------
/packages/web-ui/src/pages/__tests__/__snapshots__/JobPage.test.jsx.snap:
--------------------------------------------------------------------------------
1 | // Jest Snapshot v1, https://goo.gl/fbAQLP
2 |
3 | exports[` displays a job posting 1`] = `"Senior Front End Developer
Vienna43.400,00 – 58.800,00 EUR"`;
4 |
5 | exports[` displays a loading info 1`] = `"Loading…"`;
6 |
7 | exports[` displays a not found 1`] = `""`;
8 |
--------------------------------------------------------------------------------
/packages/backend-for-webui/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "backend-for-webui",
3 | "version": "1.0.0",
4 | "description": "the back end for front end",
5 | "main": "index.js",
6 | "scripts": {
7 | "test": "jest --setupFiles ./pact.setup.js",
8 | "start": "node src/index.js"
9 | },
10 | "jest": {
11 | "testEnvironment": "node"
12 | },
13 | "author": "Mateusz Sokola",
14 | "license": "MIT",
15 | "dependencies": {
16 | "axios": "^0.17.1",
17 | "cors": "^2.8.4",
18 | "express": "^4.16.2",
19 | "lodash": "^4.17.4",
20 | "querystring": "^0.2.0"
21 | },
22 | "devDependencies": {
23 | "@pact-foundation/pact": "^5.5.0",
24 | "jest": "^22.1.4"
25 | }
26 | }
27 |
--------------------------------------------------------------------------------
/packages/job-service/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "job-service",
3 | "version": "1.0.0",
4 | "description": "a job micro-service",
5 | "main": "index.js",
6 | "scripts": {
7 | "test:provider": "pact verify -u $(pwd)/../../pacts/backend-for-webui-job-service.json -b http://localhost:3001 --provider-states-setup-url=http://localhost:3001/setup",
8 | "dev": "cross-env NODE_ENV=development node src/index.js",
9 | "start": "node src/index.js"
10 | },
11 | "author": "Mateusz Sokola",
12 | "license": "MIT",
13 | "dependencies": {
14 | "express": "^4.16.2",
15 | "lodash": "^4.17.4",
16 | "querystring": "^0.2.0"
17 | },
18 | "devDependencies": {
19 | "@pact-foundation/pact-node": "^6.11.0",
20 | "cross-env": "^5.1.3"
21 | }
22 | }
23 |
--------------------------------------------------------------------------------
/packages/job-service/src/models/JobModel.js:
--------------------------------------------------------------------------------
1 | const generateJob = require('../utils/jobGenerator');
2 |
3 | const AMOUNT_OF_JOBS = 2000;
4 |
5 | // generates 2000 random job offers, sorted by date.
6 | let jobs = [...Array(AMOUNT_OF_JOBS)].map(() => generateJob()).sort((jobA, jobB) => {
7 | const dateA = new Date(jobA.createdAt);
8 | const dateB = new Date(jobB.createdAt);
9 |
10 | return dateB.getTime() - dateA.getTime();
11 | });
12 |
13 | function findJob(id, slug) {
14 | return jobs.filter((j) => id === j.id && slug === j.slug);
15 | }
16 |
17 | function getJobList(skip = 0, amount = 100) {
18 | const offset = skip + amount;
19 | return jobs.slice(skip, offset);
20 | }
21 |
22 | function setupJobList(newJobs) {
23 | jobs = newJobs;
24 | }
25 |
26 | module.exports.findJob = findJob;
27 | module.exports.getJobList = getJobList;
28 | module.exports.setupJobList = setupJobList;
29 |
--------------------------------------------------------------------------------
/packages/job-service/src/utils/jobGenerator/generateJob.js:
--------------------------------------------------------------------------------
1 | const { random } = require('lodash');
2 |
3 | const JobId = require('./jobId');
4 | const generateSalaryRange = require('./generateSalaryRange');
5 | const generateJobTitle = require('./generateTitle');
6 |
7 | function generateDate() {
8 | return new Date(Date.now() - random(1, 1000 * 3600 * 24 * 15)).toString()
9 | }
10 |
11 | function titleToSlug(title) {
12 | return title.split(' ')
13 | .map((word) => word.toLocaleLowerCase())
14 | .join('-');
15 | }
16 |
17 | function generateJob() {
18 | const title = generateJobTitle();
19 | const salary = generateSalaryRange();
20 |
21 | return {
22 | id: JobId().nextId(),
23 | slug: titleToSlug(title),
24 | title,
25 | createdAt: generateDate(),
26 | salaryFrom: salary.from,
27 | salaryTo: salary.to,
28 | city: 'Vienna',
29 | };
30 | }
31 |
32 | module.exports = generateJob;
--------------------------------------------------------------------------------
/packages/web-ui/src/index.jsx:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import { render } from 'react-dom';
3 | import { HashRouter, Link, Route } from 'react-router-dom';
4 |
5 | import NotFound from "./components/NotFound";
6 | import JobListPage from "./pages/JobListPage";
7 | import JobPage from "./pages/JobPage";
8 |
9 | // Load styles
10 | require('./style.scss');
11 |
12 | render(
13 |
14 |
15 |
22 |
23 |
24 |
25 | ,
26 | document.getElementById('web-ui')
27 | );
--------------------------------------------------------------------------------
/packages/job-service/src/utils/jobGenerator/generateTitle.js:
--------------------------------------------------------------------------------
1 | const { random } = require('lodash');
2 |
3 | const TITLE_LIST = [
4 | 'UI',
5 | 'Software',
6 | 'Front End',
7 | 'Back End',
8 | 'JavaScript',
9 | 'PHP',
10 | 'Java',
11 | 'React',
12 | 'Support',
13 | 'Full Stack'
14 | ];
15 | const PREFIX_LIST = [
16 | 'Junior',
17 | 'Regular',
18 | 'Senior',
19 | 'Principal',
20 | ''
21 | ]
22 | const POSTFIX_LIST = [
23 | 'Engineer',
24 | 'Developer',
25 | 'Architect',
26 | 'Manager'
27 | ]
28 |
29 | function generateJobTitle() {
30 | const title = [
31 | PREFIX_LIST[random(PREFIX_LIST.length - 1)],
32 | TITLE_LIST[random(TITLE_LIST.length - 1)],
33 | POSTFIX_LIST[random(POSTFIX_LIST.length - 1)]
34 | ];
35 | return title // ... => [ '', 'UI', 'Manager' ]
36 | .join(' ') // [ '', 'UI', 'Manager' ] => ' UI Manager'
37 | .trim(); // ' UI Manager' => 'UI Manager'
38 | }
39 |
40 | module.exports = generateJobTitle;
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2018 Mateusz Sokola
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/packages/web-ui/src/style.scss:
--------------------------------------------------------------------------------
1 | @import "~@blueprintjs/core/dist/blueprint.css";
2 | @import "~@blueprintjs/core/src/common/variables";
3 |
4 | body {
5 | background: $light-gray5;
6 | margin: 0;
7 | }
8 |
9 | .pt-job-list {
10 | margin: 0;
11 | padding: 0;
12 | list-style: none;
13 |
14 | li {
15 | padding: $pt-grid-size $pt-grid-size * 1.5;
16 |
17 | &:nth-child(odd) {
18 | background: rgba($gray5, 0.2);
19 | }
20 |
21 | &:last-child {
22 | margin-bottom: 0;
23 | }
24 |
25 | a {
26 | display: flex;
27 | align-items: center;
28 | flex-flow: row nowrap;
29 | justify-content: space-between;
30 | color: $dark-gray5;
31 |
32 | &:hover {
33 | text-decoration: none;
34 | }
35 |
36 | & > span {
37 |
38 | &:first-child {
39 | width: 70%;
40 | }
41 | }
42 | }
43 |
44 | .title {
45 | font-weight: bold;
46 | display: block;
47 | }
48 | }
49 | }
--------------------------------------------------------------------------------
/packages/web-ui/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "web-ui",
3 | "version": "1.0.0",
4 | "description": "web-ui",
5 | "main": "index.js",
6 | "scripts": {
7 | "build": "webpack",
8 | "clean": "rm -rf target",
9 | "test": "jest --setupFiles ./pact.setup.js ./enzyme.setup.js",
10 | "start": "webpack-dev-server --progress --colors"
11 | },
12 | "author": "Mateusz Sokola",
13 | "license": "MIT",
14 | "dependencies": {
15 | "axios": "^0.17.1",
16 | "express": "^4.16.2",
17 | "moment": "^2.20.1",
18 | "react": "^16.2.0",
19 | "react-dom": "^16.2.0",
20 | "react-moment": "^0.7.0",
21 | "react-router-dom": "^4.2.2"
22 | },
23 | "devDependencies": {
24 | "@babel/core": "^7.0.0-beta.40",
25 | "@babel/preset-env": "^7.0.0-beta.39",
26 | "@babel/preset-react": "^7.0.0-beta.39",
27 | "@blueprintjs/core": "^1.35.4",
28 | "@pact-foundation/pact": "^5.5.0",
29 | "babel-core": "^7.0.0-bridge.0",
30 | "babel-jest": "^22.2.2",
31 | "babel-loader": "^8.0.0-beta.0",
32 | "css-loader": "^0.28.9",
33 | "enzyme": "^3.3.0",
34 | "enzyme-adapter-react-16": "^1.1.1",
35 | "extract-text-webpack-plugin": "^3.0.2",
36 | "file-loader": "^1.1.6",
37 | "html-webpack-plugin": "^2.30.1",
38 | "jest": "^22.1.4",
39 | "node-sass": "^4.7.2",
40 | "querystring": "^0.2.0",
41 | "react-transition-group": "^2.2.1",
42 | "regenerator-runtime": "^0.11.1",
43 | "sass-loader": "^6.0.6",
44 | "source-map-loader": "^0.2.3",
45 | "style-loader": "^0.20.1",
46 | "url-loader": "^0.6.2",
47 | "webpack": "^3.11.0",
48 | "webpack-dev-server": "^2.11.1"
49 | }
50 | }
51 |
--------------------------------------------------------------------------------
/packages/web-ui/src/components/Salary.jsx:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import ReactDOM from 'react-dom';
3 | import PropTypes from 'prop-types';
4 |
5 | const Salary = ({ from, to }) => {
6 | function getCents(amount) {
7 | return amount % 100;
8 | }
9 |
10 | function getDollars(amount) {
11 | return amount / 100;
12 | }
13 |
14 | function reverseString(string) {
15 | return `${string}`.split('').reverse().join('');
16 | }
17 |
18 | function addSeparators(string) {
19 | return `${string}`.match(/[0-9]{1,3}/g).join('.');
20 | }
21 |
22 | function compose(...funcs) {
23 | return funcs.reduce(function (f, g) {
24 | return function (...args) {
25 | return f(g(...args));
26 | };
27 | });
28 | }
29 |
30 | function formatCents(cents) {
31 | if (cents < 10) {
32 | return `0${cents}`;
33 | }
34 |
35 | return `${cents}`;
36 | }
37 |
38 | function formatMoney(amountInCents) {
39 | const cents = getCents(amountInCents);
40 | const dollars = amountInCents - cents; // dollars = total amount - cents
41 |
42 | return formatDollars(dollars) + ',' + formatCents(cents);
43 | }
44 |
45 | // executed from the end: getDollars => reverseString => addSeparators => reverseString
46 | const formatDollars = compose(reverseString, addSeparators, reverseString, getDollars);
47 |
48 | return {formatMoney(from)} – {formatMoney(to)} EUR;
49 | }
50 |
51 | Salary.propTypes = {
52 | from: PropTypes.number.isRequired,
53 | to: PropTypes.number.isRequired,
54 | }
55 |
56 | export default Salary;
--------------------------------------------------------------------------------
/packages/web-ui/src/pages/JobListPage.jsx:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import Moment from 'react-moment';
3 | import { Link } from 'react-router-dom';
4 |
5 | import fetchJobList from '../services/fetchJobList.js';
6 | import Loading from '../components/Loading';
7 | import Salary from '../components/Salary';
8 |
9 | class JobListPage extends React.Component {
10 | constructor() {
11 | super();
12 | this.state = {
13 | loading: false,
14 | error: null,
15 | items: []
16 | }
17 | }
18 |
19 | componentWillMount() {
20 | this.setState({ loading: true, error: null });
21 |
22 | fetchJobList()
23 | .then(items => {
24 | this.setState({ loading: false, error: null, items });
25 | })
26 | .catch(error => {
27 | this.setState({ loading: false, error: error.message });
28 | })
29 | }
30 |
31 | render() {
32 | const { loading, error, items } = this.state;
33 |
34 | if (loading) {
35 | return ;
36 | }
37 |
38 | return (
39 |
40 | {items.map((item, i) => {
41 | const dateToFormat = (new Date(item.createdAt));
42 | return (-
43 |
44 |
45 | {item.title}
46 | {item.city}
47 |
48 |
49 |
50 |
51 |
)
52 | })}
53 |
54 | )
55 | }
56 | }
57 |
58 | export default JobListPage;
--------------------------------------------------------------------------------
/packages/web-ui/src/pages/JobPage.jsx:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import Moment from 'react-moment';
3 | import { Link } from 'react-router-dom';
4 |
5 | import fetchJob from '../services/fetchJob.js';
6 | import NotFound from '../components/NotFound';
7 | import Loading from '../components/Loading';
8 | import Salary from '../components/Salary';
9 |
10 | class JobPage extends React.Component {
11 | constructor(props) {
12 | super(props);
13 | this.state = {
14 | loading: false,
15 | error: null,
16 | item: null
17 | }
18 | }
19 |
20 | componentWillMount() {
21 | const { id, slug } = this.props.match.params;
22 | this.setState({ loading: true, error: null });
23 |
24 | fetchJob({ id, slug })
25 | .then(item => {
26 | this.setState({ loading: false, error: null, item });
27 | })
28 | .catch(error => {
29 | this.setState({ loading: false, error: error.message });
30 | })
31 | }
32 |
33 | render() {
34 | const { loading, error, item } = this.state;
35 |
36 | if (loading) {
37 | return ;
38 | }
39 |
40 | if (error) {
41 | return ;
42 | }
43 |
44 | const dateToFormat = (new Date(item.createdAt));
45 |
46 | return (
47 |
48 |
{item.title}
49 | {item.city}
50 |
51 |
52 |
53 | )
54 | }
55 | }
56 |
57 | export default JobPage;
--------------------------------------------------------------------------------
/packages/web-ui/src/pages/__tests__/__snapshots__/JobListPage.test.jsx.snap:
--------------------------------------------------------------------------------
1 | // Jest Snapshot v1, https://goo.gl/fbAQLP
2 |
3 | exports[` displays a job listing 1`] = `"- Senior Front End DeveloperVienna43.400,00 – 58.800,00 EUR
- Senior PHP DeveloperVienna49.000,00 – 65.800,00 EUR
- Back End ArchitectVienna40.600,00 – 65.800,00 EUR
- UI DeveloperVienna40.600,00 – 67.200,00 EUR
- Principal JavaScript ManagerVienna36.400,00 – 61.600,00 EUR
- Regular JavaScript EngineerVienna40.600,00 – 56.000,00 EUR
"`;
4 |
5 | exports[` displays a loading screen 1`] = `"Loading…"`;
6 |
--------------------------------------------------------------------------------
/packages/web-ui/webpack.config.js:
--------------------------------------------------------------------------------
1 | const ExtractTextPlugin = require('extract-text-webpack-plugin')
2 | const HtmlWebpackPlugin = require('html-webpack-plugin')
3 | const path = require('path')
4 |
5 | const webpack = {
6 | entry: './src/index.jsx',
7 | output: {
8 | path: path.resolve(__dirname, 'target'),
9 | filename: 'bundle.js',
10 | },
11 | devtool: 'source-map',
12 | resolve: {
13 | extensions: ['.jsx', '.js', '.json', '.css', '.scss'],
14 | },
15 | devServer: {
16 | contentBase: false,
17 | compress: true,
18 | port: 9000
19 | },
20 | module: {
21 | loaders: [
22 | {
23 | test: /.jsx?$/,
24 | loader: 'babel-loader',
25 | exclude: /node_modules/,
26 | }, {
27 | test: /\.js$/,
28 | use: ['source-map-loader'],
29 | enforce: 'pre',
30 | }, {
31 | test: /\.scss$/,
32 | loader: ExtractTextPlugin.extract(({
33 | fallback: 'style-loader',
34 | use: [
35 | 'css-loader',
36 | 'sass-loader'
37 | ]
38 | }))
39 | }, {
40 | test: /\.(woff|woff2)$/,
41 | use: {
42 | loader: 'url-loader',
43 | options: {
44 | name: 'static/fonts/[hash].[ext]',
45 | limit: 5000,
46 | mimetype: 'application/font-woff'
47 | }
48 | }
49 | }, {
50 | test: /\.(ttf|eot|svg)$/,
51 | use: {
52 | loader: 'file-loader',
53 | options: {
54 | name: 'static/fonts/[hash].[ext]'
55 | }
56 | }
57 | }
58 | ]
59 | },
60 | plugins: [
61 | new ExtractTextPlugin('bundle.css'),
62 | new HtmlWebpackPlugin({
63 | filename: 'index.html',
64 | template: 'src/index.html'
65 | })
66 | ]
67 | }
68 |
69 | module.exports = webpack
--------------------------------------------------------------------------------
/packages/backend-for-webui/src/controllers/Job.js:
--------------------------------------------------------------------------------
1 | const axios = require('axios');
2 | const getQueryParams = require('../utils/getQueryParams');
3 |
4 | const serviceUrl = 'http://localhost:3001';
5 | const endpoint = '/v1';
6 | const timeout = 1000;
7 |
8 | async function retrieveJob(req, res) {
9 | try {
10 | const id = parseInt(req.params.id, 10);
11 | const slug = req.params.slug;
12 |
13 | // example: http://localhost:3001/v1/1-senior-back-end-developer
14 | const url = `${serviceUrl}${endpoint}/${id}-${slug}`;
15 | const response = await axios.get(url, { timeout });
16 | const item = response.data;
17 |
18 | return res.json(item);
19 | }
20 | catch (error) {
21 | if (error.response) {
22 | return res.status(error.response.status).json({ status: 'NOT_FOUND' })
23 | }
24 | console.log('Error', error.message);
25 |
26 | return res.status(500).json({ status: 'ERROR' })
27 | }
28 | }
29 |
30 | async function retrieveList(req, res) {
31 | try {
32 | const params = getQueryParams(req.url);
33 | const skip = parseInt(params.skip, 10) || 0;
34 | const amount = parseInt(params.amount, 10) || 25;
35 |
36 | // example: http://localhost:3001/v1
37 | const url = `${serviceUrl}${endpoint}`;
38 | const response = await axios.get(url, {
39 | params: {
40 | skip,
41 | amount
42 | },
43 | timeout
44 | });
45 |
46 | const items = response.data;
47 | return res.json(items)
48 | }
49 | catch (error) {
50 | if (error.response) {
51 | return res.status(error.response.status).json({ status: 'NOT_FOUND' })
52 | }
53 | console.log('Error', error.message);
54 |
55 | return res.status(500).json({ status: 'ERROR' })
56 | }
57 | }
58 |
59 | module.exports.retrieveJob = retrieveJob;
60 | module.exports.retrieveList = retrieveList;
--------------------------------------------------------------------------------
/packages/web-ui/src/pages/__tests__/JobPage.test.jsx:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import { mount } from 'enzyme';
3 |
4 | import JobPage from '../JobPage';
5 |
6 | jest.mock('react-router-dom')
7 |
8 | const EXPECTED_BODY = {
9 | id: 1,
10 | slug: 'senior-front-end-developer',
11 | title: 'Senior Front End Developer',
12 | createdAt: 'Sat Feb 10 2018 18:12:40 GMT+0100 (CET)',
13 | salaryFrom: 4340000,
14 | salaryTo: 5880000,
15 | city: 'Vienna'
16 | };
17 |
18 | describe('', () => {
19 | it('displays a job posting', () => {
20 | const match = {
21 | params: {
22 | id: 1,
23 | slug: 'senior-front-end-developer'
24 | }
25 | };
26 | const wrapper = mount();
27 |
28 | wrapper.setState({
29 | loading: false,
30 | error: null,
31 | item: EXPECTED_BODY
32 | })
33 |
34 | expect(wrapper.html()).toMatchSnapshot();
35 | });
36 |
37 | it('displays a not found', () => {
38 | const match = {
39 | params: {
40 | id: 2,
41 | slug: 'non-existent-position'
42 | }
43 | };
44 | const wrapper = mount();
45 |
46 | wrapper.setState({
47 | loading: false,
48 | error: new Error('Not Found'),
49 | item: null
50 | })
51 |
52 | expect(wrapper.html()).toMatchSnapshot();
53 | });
54 |
55 | it('displays a loading info', () => {
56 | const match = {
57 | params: {
58 | id: 1,
59 | slug: 'senior-front-end-developer'
60 | }
61 | };
62 | const wrapper = mount();
63 |
64 | wrapper.setState({
65 | loading: true,
66 | error: null,
67 | item: null
68 | })
69 |
70 | expect(wrapper.html()).toMatchSnapshot();
71 | });
72 | })
--------------------------------------------------------------------------------
/packages/web-ui/src/pages/__tests__/JobListPage.test.jsx:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import { mount } from 'enzyme';
3 |
4 | import JobListPage from '../JobListPage';
5 |
6 | jest.mock('react-router-dom')
7 |
8 | const EXPECTED_BODY = [
9 | {
10 | id: 1,
11 | slug: 'senior-front-end-developer',
12 | title: 'Senior Front End Developer',
13 | createdAt: 'Sat Feb 10 2018 18:12:40 GMT+0100 (CET)',
14 | salaryFrom: 4340000,
15 | salaryTo: 5880000,
16 | city: 'Vienna'
17 | }, {
18 | id: 2,
19 | slug: 'senior-php-developer',
20 | title: 'Senior PHP Developer',
21 | createdAt: 'Sat Feb 17 2018 14:38:50 GMT+0100 (CET)',
22 | salaryFrom: 4900000,
23 | salaryTo: 6580000,
24 | city: 'Vienna'
25 | }, {
26 | id: 3,
27 | slug: 'back-end-architect',
28 | title: 'Back End Architect',
29 | createdAt: 'Sat Feb 17 2018 14:22:39 GMT+0100 (CET)',
30 | salaryFrom: 4060000,
31 | salaryTo: 6580000,
32 | city: 'Vienna'
33 | }, {
34 | id: 4,
35 | slug: 'ui-developer',
36 | title: 'UI Developer',
37 | createdAt: 'Sat Feb 17 2018 14:07:15 GMT+0100 (CET)',
38 | salaryFrom: 4060000,
39 | salaryTo: 6720000,
40 | city: 'Vienna'
41 | }, {
42 | id: 5,
43 | slug: 'principal-javascript-manager',
44 | title: 'Principal JavaScript Manager',
45 | createdAt: 'Sat Feb 17 2018 14:02:13 GMT+0100 (CET)',
46 | salaryFrom: 3640000,
47 | salaryTo: 6160000,
48 | city: 'Vienna'
49 | }, {
50 | id: 6,
51 | slug: 'regular-javascript-engineer',
52 | title: 'Regular JavaScript Engineer',
53 | createdAt: 'Sat Feb 17 2018 13:53:12 GMT+0100 (CET)',
54 | salaryFrom: 4060000,
55 | salaryTo: 5600000,
56 | city: 'Vienna'
57 | }
58 | ];
59 |
60 | describe.skip('', () => {
61 | it('displays a loading screen', () => {
62 | const wrapper = mount();
63 |
64 | expect(wrapper.html()).toMatchSnapshot();
65 | });
66 |
67 | it('displays a job listing', () => {
68 | const wrapper = mount();
69 |
70 | wrapper.setState({
71 | loading: false,
72 | error: null,
73 | items: EXPECTED_BODY
74 | })
75 |
76 | expect(wrapper.html()).toMatchSnapshot();
77 | });
78 |
79 | })
--------------------------------------------------------------------------------
/packages/job-service/src/controllers/Job.js:
--------------------------------------------------------------------------------
1 | const { findJob, getJobList, setupJobList } = require('../models/JobModel');
2 | const getQueryParams = require('../utils/getQueryParams');
3 |
4 | function retrieveJob(req, res) {
5 | const id = parseInt(req.params.id, 10);
6 | const slug = req.params.slug;
7 |
8 | const items = findJob(id, slug);
9 |
10 | if (items.length) {
11 | return res.json(items[0])
12 | }
13 |
14 | return res.status(404).json({ status: 'NOT_FOUND' })
15 | }
16 |
17 | function retrieveJobList(req, res) {
18 | const params = getQueryParams(req.url);
19 | const skip = parseInt(params.skip, 10) || 0;
20 | const amount = parseInt(params.amount, 10) || 25;
21 | const jobs = getJobList(skip, amount);
22 |
23 | res.status(200).json(jobs);
24 | }
25 |
26 | function setupTestData(req, res) {
27 | setupJobList([
28 | {
29 | id: 1,
30 | slug: 'senior-front-end-developer',
31 | title: 'Senior Front End Developer',
32 | createdAt: 'Sat Feb 10 2018 18:12:40 GMT+0100 (CET)',
33 | salaryFrom: 4340000,
34 | salaryTo: 5880000,
35 | city: 'Vienna'
36 | },{
37 | id: 2,
38 | slug: 'senior-php-developer',
39 | title: 'Senior PHP Developer',
40 | createdAt: 'Sat Feb 17 2018 14:38:50 GMT+0100 (CET)',
41 | salaryFrom: 4900000,
42 | salaryTo: 6580000,
43 | city: 'Vienna'
44 | },{
45 | id: 3,
46 | slug: 'back-end-architect',
47 | title: 'Back End Architect',
48 | createdAt: 'Sat Feb 17 2018 14:22:39 GMT+0100 (CET)',
49 | salaryFrom: 4060000,
50 | salaryTo: 6580000,
51 | city: 'Vienna'
52 | },{
53 | id: 4,
54 | slug: 'ui-developer',
55 | title: 'UI Developer',
56 | createdAt: 'Sat Feb 17 2018 14:07:15 GMT+0100 (CET)',
57 | salaryFrom: 4060000,
58 | salaryTo: 6720000,
59 | city: 'Vienna'
60 | },{
61 | id: 5,
62 | slug: 'principal-javascript-manager',
63 | title: 'Principal JavaScript Manager',
64 | createdAt: 'Sat Feb 17 2018 14:02:13 GMT+0100 (CET)',
65 | salaryFrom: 3640000,
66 | salaryTo: 6160000,
67 | city: 'Vienna'
68 | },{
69 | id: 6,
70 | slug: 'regular-javascript-engineer',
71 | title: 'Regular JavaScript Engineer',
72 | createdAt: 'Sat Feb 17 2018 13:53:12 GMT+0100 (CET)',
73 | salaryFrom: 4060000,
74 | salaryTo: 5600000,
75 | city: 'Vienna'
76 | }
77 | ]);
78 |
79 | return res.status(200).send();
80 | }
81 |
82 | module.exports.setupTestData = setupTestData;
83 | module.exports.retrieveJob = retrieveJob;
84 | module.exports.retrieveJobList = retrieveJobList;
--------------------------------------------------------------------------------
/pacts/web-ui-backend-for-webui.json:
--------------------------------------------------------------------------------
1 | {
2 | "consumer": {
3 | "name": "web-ui"
4 | },
5 | "provider": {
6 | "name": "backend-for-webui"
7 | },
8 | "interactions": [
9 | {
10 | "description": "a request for an job offer with id 1 and slug senior-front-end-developer",
11 | "providerState": "has a job offer",
12 | "request": {
13 | "method": "GET",
14 | "path": "/v1/job/1-senior-front-end-developer",
15 | "headers": {
16 | "Accept": "application/json"
17 | },
18 | "matchingRules": {
19 | "$.headers.Accept": {
20 | "match": "type"
21 | }
22 | }
23 | },
24 | "response": {
25 | "status": 200,
26 | "headers": {
27 | "Content-Type": "application/json"
28 | },
29 | "body": {
30 | "id": 1,
31 | "slug": "senior-front-end-developer",
32 | "title": "Senior Front End Developer",
33 | "createdAt": "Sat Feb 10 2018 18:12:40 GMT+0100 (CET)",
34 | "salaryFrom": 4340000,
35 | "salaryTo": 5880000,
36 | "city": "Vienna"
37 | },
38 | "matchingRules": {
39 | "$.headers.Content-Type": {
40 | "match": "type"
41 | }
42 | }
43 | }
44 | },
45 | {
46 | "description": "a request for non existent job",
47 | "providerState": "has no job offer",
48 | "request": {
49 | "method": "GET",
50 | "path": "/v1/job/2-non-existent-offer",
51 | "headers": {
52 | "Accept": "application/json"
53 | },
54 | "matchingRules": {
55 | "$.headers.Accept": {
56 | "match": "type"
57 | }
58 | }
59 | },
60 | "response": {
61 | "status": 404,
62 | "headers": {
63 | "Content-Type": "application/json"
64 | },
65 | "body": {
66 | "status": "NOT_FOUND"
67 | },
68 | "matchingRules": {
69 | "$.headers.Content-Type": {
70 | "match": "type"
71 | }
72 | }
73 | }
74 | },
75 | {
76 | "description": "a request for the job offers",
77 | "providerState": "has some job offers",
78 | "request": {
79 | "method": "GET",
80 | "path": "/v1/jobs",
81 | "headers": {
82 | "Accept": "application/json"
83 | },
84 | "matchingRules": {
85 | "$.headers.Accept": {
86 | "match": "type"
87 | }
88 | }
89 | },
90 | "response": {
91 | "status": 200,
92 | "headers": {
93 | "Content-Type": "application/json"
94 | },
95 | "body": [
96 | {
97 | "id": 1,
98 | "slug": "senior-front-end-developer",
99 | "title": "Senior Front End Developer",
100 | "createdAt": "Sat Feb 10 2018 18:12:40 GMT+0100 (CET)",
101 | "salaryFrom": 4340000,
102 | "salaryTo": 5880000,
103 | "city": "Vienna"
104 | },
105 | {
106 | "id": 2,
107 | "slug": "senior-php-developer",
108 | "title": "Senior PHP Developer",
109 | "createdAt": "Sat Feb 17 2018 14:38:50 GMT+0100 (CET)",
110 | "salaryFrom": 4900000,
111 | "salaryTo": 6580000,
112 | "city": "Vienna"
113 | },
114 | {
115 | "id": 3,
116 | "slug": "back-end-architect",
117 | "title": "Back End Architect",
118 | "createdAt": "Sat Feb 17 2018 14:22:39 GMT+0100 (CET)",
119 | "salaryFrom": 4060000,
120 | "salaryTo": 6580000,
121 | "city": "Vienna"
122 | },
123 | {
124 | "id": 4,
125 | "slug": "ui-developer",
126 | "title": "UI Developer",
127 | "createdAt": "Sat Feb 17 2018 14:07:15 GMT+0100 (CET)",
128 | "salaryFrom": 4060000,
129 | "salaryTo": 6720000,
130 | "city": "Vienna"
131 | },
132 | {
133 | "id": 5,
134 | "slug": "principal-javascript-manager",
135 | "title": "Principal JavaScript Manager",
136 | "createdAt": "Sat Feb 17 2018 14:02:13 GMT+0100 (CET)",
137 | "salaryFrom": 3640000,
138 | "salaryTo": 6160000,
139 | "city": "Vienna"
140 | },
141 | {
142 | "id": 6,
143 | "slug": "regular-javascript-engineer",
144 | "title": "Regular JavaScript Engineer",
145 | "createdAt": "Sat Feb 17 2018 13:53:12 GMT+0100 (CET)",
146 | "salaryFrom": 4060000,
147 | "salaryTo": 5600000,
148 | "city": "Vienna"
149 | }
150 | ],
151 | "matchingRules": {
152 | "$.headers.Content-Type": {
153 | "match": "type"
154 | }
155 | }
156 | }
157 | }
158 | ],
159 | "metadata": {
160 | "pactSpecification": {
161 | "version": "2.0.0"
162 | }
163 | }
164 | }
--------------------------------------------------------------------------------
/pacts/backend-for-webui-job-service.json:
--------------------------------------------------------------------------------
1 | {
2 | "consumer": {
3 | "name": "backend-for-webui"
4 | },
5 | "provider": {
6 | "name": "job-service"
7 | },
8 | "interactions": [
9 | {
10 | "description": "a request for an job offer with id 1 and slug senior-front-end-developer",
11 | "providerState": "has a requested job offer",
12 | "request": {
13 | "method": "GET",
14 | "path": "/v1/1-senior-front-end-developer",
15 | "headers": {
16 | "Accept": "application/json"
17 | },
18 | "matchingRules": {
19 | "$.headers.Accept": {
20 | "match": "type"
21 | }
22 | }
23 | },
24 | "response": {
25 | "status": 200,
26 | "headers": {
27 | "Content-Type": "application/json"
28 | },
29 | "body": {
30 | "id": 1,
31 | "slug": "senior-front-end-developer",
32 | "title": "Senior Front End Developer",
33 | "createdAt": "Sat Feb 10 2018 18:12:40 GMT+0100 (CET)",
34 | "salaryFrom": 4340000,
35 | "salaryTo": 5880000,
36 | "city": "Vienna"
37 | },
38 | "matchingRules": {
39 | "$.headers.Content-Type": {
40 | "match": "type"
41 | }
42 | }
43 | }
44 | },
45 | {
46 | "description": "a request for a job offer",
47 | "providerState": "has no offer matching given criteria",
48 | "request": {
49 | "method": "GET",
50 | "path": "/v1/2-non-existent-offer",
51 | "headers": {
52 | "Accept": "application/json"
53 | },
54 | "matchingRules": {
55 | "$.headers.Accept": {
56 | "match": "type"
57 | }
58 | }
59 | },
60 | "response": {
61 | "status": 404,
62 | "headers": {
63 | "Content-Type": "application/json"
64 | },
65 | "body": {
66 | "status": "NOT_FOUND"
67 | },
68 | "matchingRules": {
69 | "$.headers.Content-Type": {
70 | "match": "type"
71 | }
72 | }
73 | }
74 | },
75 | {
76 | "description": "a request for the first 5 job offers",
77 | "providerState": "has 5 job offers",
78 | "request": {
79 | "method": "GET",
80 | "path": "/v1",
81 | "query": "skip=0&amount=5",
82 | "headers": {
83 | "Accept": "application/json"
84 | },
85 | "matchingRules": {
86 | "$.headers.Accept": {
87 | "match": "type"
88 | }
89 | }
90 | },
91 | "response": {
92 | "status": 200,
93 | "headers": {
94 | "Content-Type": "application/json"
95 | },
96 | "body": [
97 | {
98 | "id": 1,
99 | "slug": "senior-front-end-developer",
100 | "title": "Senior Front End Developer",
101 | "createdAt": "Sat Feb 10 2018 18:12:40 GMT+0100 (CET)",
102 | "salaryFrom": 4340000,
103 | "salaryTo": 5880000,
104 | "city": "Vienna"
105 | },
106 | {
107 | "id": 2,
108 | "slug": "senior-php-developer",
109 | "title": "Senior PHP Developer",
110 | "createdAt": "Sat Feb 17 2018 14:38:50 GMT+0100 (CET)",
111 | "salaryFrom": 4900000,
112 | "salaryTo": 6580000,
113 | "city": "Vienna"
114 | },
115 | {
116 | "id": 3,
117 | "slug": "back-end-architect",
118 | "title": "Back End Architect",
119 | "createdAt": "Sat Feb 17 2018 14:22:39 GMT+0100 (CET)",
120 | "salaryFrom": 4060000,
121 | "salaryTo": 6580000,
122 | "city": "Vienna"
123 | },
124 | {
125 | "id": 4,
126 | "slug": "ui-developer",
127 | "title": "UI Developer",
128 | "createdAt": "Sat Feb 17 2018 14:07:15 GMT+0100 (CET)",
129 | "salaryFrom": 4060000,
130 | "salaryTo": 6720000,
131 | "city": "Vienna"
132 | },
133 | {
134 | "id": 5,
135 | "slug": "principal-javascript-manager",
136 | "title": "Principal JavaScript Manager",
137 | "createdAt": "Sat Feb 17 2018 14:02:13 GMT+0100 (CET)",
138 | "salaryFrom": 3640000,
139 | "salaryTo": 6160000,
140 | "city": "Vienna"
141 | }
142 | ],
143 | "matchingRules": {
144 | "$.headers.Content-Type": {
145 | "match": "type"
146 | }
147 | }
148 | }
149 | },
150 | {
151 | "description": "a request for 5 job offers skipping first 5",
152 | "providerState": "has 1 more job offer",
153 | "request": {
154 | "method": "GET",
155 | "path": "/v1",
156 | "query": "skip=5&amount=5",
157 | "headers": {
158 | "Accept": "application/json"
159 | },
160 | "matchingRules": {
161 | "$.headers.Accept": {
162 | "match": "type"
163 | }
164 | }
165 | },
166 | "response": {
167 | "status": 200,
168 | "headers": {
169 | "Content-Type": "application/json"
170 | },
171 | "body": [
172 | {
173 | "id": 6,
174 | "slug": "regular-javascript-engineer",
175 | "title": "Regular JavaScript Engineer",
176 | "createdAt": "Sat Feb 17 2018 13:53:12 GMT+0100 (CET)",
177 | "salaryFrom": 4060000,
178 | "salaryTo": 5600000,
179 | "city": "Vienna"
180 | }
181 | ],
182 | "matchingRules": {
183 | "$.headers.Content-Type": {
184 | "match": "type"
185 | }
186 | }
187 | }
188 | }
189 | ],
190 | "metadata": {
191 | "pactSpecification": {
192 | "version": "2.0.0"
193 | }
194 | }
195 | }
--------------------------------------------------------------------------------
/packages/web-ui/src/services/__tests__/services.test.js:
--------------------------------------------------------------------------------
1 | import { Matchers } from '@pact-foundation/pact';
2 |
3 | import fetchJob from '../fetchJob';
4 | import fetchJobList from '../fetchJobList';
5 |
6 | // it takes about 4 sec to spin up the mock server
7 | jest.setTimeout(20000);
8 |
9 | describe('Services', () => {
10 | beforeAll(() => provider.setup());
11 |
12 | afterAll(() => provider.finalize());
13 |
14 | describe('fetchJob', () => {
15 | const OFFER_BODY = {
16 | id: 1,
17 | slug: 'senior-front-end-developer',
18 | title: 'Senior Front End Developer',
19 | createdAt: 'Sat Feb 10 2018 18:12:40 GMT+0100 (CET)',
20 | salaryFrom: 4340000,
21 | salaryTo: 5880000,
22 | city: 'Vienna'
23 | };
24 |
25 | beforeAll(() => provider.addInteraction({
26 | state: 'has a job offer',
27 | uponReceiving: 'a request for an job offer with id 1 and slug senior-front-end-developer',
28 | withRequest: {
29 | method: 'GET',
30 | // endpoint on the back-end for web-ui side
31 | path: '/v1/job/1-senior-front-end-developer',
32 | headers: {
33 | 'Accept': Matchers.somethingLike('application/json')
34 | }
35 | },
36 | willRespondWith: {
37 | status: 200,
38 | headers: {
39 | 'Content-Type': Matchers.somethingLike('application/json')
40 | },
41 | body: OFFER_BODY
42 | }
43 | }));
44 |
45 | beforeAll(() => provider.addInteraction({
46 | state: 'has no job offer',
47 | uponReceiving: 'a request for non existent job',
48 | withRequest: {
49 | method: 'GET',
50 | // endpoint on the back-end for web-ui side
51 | path: '/v1/job/2-non-existent-offer',
52 | headers: {
53 | 'Accept': Matchers.somethingLike('application/json')
54 | }
55 | },
56 | willRespondWith: {
57 | status: 404,
58 | headers: {
59 | 'Content-Type': Matchers.somethingLike('application/json')
60 | },
61 | body: {
62 | status: 'NOT_FOUND'
63 | }
64 | }
65 | }));
66 |
67 | it('returns a job offer', async () => {
68 | const response = await fetchJob({ id: '1', slug: 'senior-front-end-developer'});
69 | expect(response).toEqual(OFFER_BODY);
70 | });
71 |
72 | it('returns a error', async () => {
73 | try {
74 | await fetchJob({ id: '2', slug: 'non-existent-offer'});
75 | // this should never triggered
76 | expect(true).toBeFalsy();
77 | }
78 | catch (error) {
79 | const { response } = error;
80 | expect(response.status).toEqual(404);
81 | expect(response.data).toEqual({ status: 'NOT_FOUND' });
82 | }
83 | });
84 | });
85 |
86 | describe('fetchJobList', () => {
87 | const LIST_BODY = [
88 | {
89 | id: 1,
90 | slug: 'senior-front-end-developer',
91 | title: 'Senior Front End Developer',
92 | createdAt: 'Sat Feb 10 2018 18:12:40 GMT+0100 (CET)',
93 | salaryFrom: 4340000,
94 | salaryTo: 5880000,
95 | city: 'Vienna'
96 | }, {
97 | id: 2,
98 | slug: 'senior-php-developer',
99 | title: 'Senior PHP Developer',
100 | createdAt: 'Sat Feb 17 2018 14:38:50 GMT+0100 (CET)',
101 | salaryFrom: 4900000,
102 | salaryTo: 6580000,
103 | city: 'Vienna'
104 | }, {
105 | id: 3,
106 | slug: 'back-end-architect',
107 | title: 'Back End Architect',
108 | createdAt: 'Sat Feb 17 2018 14:22:39 GMT+0100 (CET)',
109 | salaryFrom: 4060000,
110 | salaryTo: 6580000,
111 | city: 'Vienna'
112 | }, {
113 | id: 4,
114 | slug: 'ui-developer',
115 | title: 'UI Developer',
116 | createdAt: 'Sat Feb 17 2018 14:07:15 GMT+0100 (CET)',
117 | salaryFrom: 4060000,
118 | salaryTo: 6720000,
119 | city: 'Vienna'
120 | }, {
121 | id: 5,
122 | slug: 'principal-javascript-manager',
123 | title: 'Principal JavaScript Manager',
124 | createdAt: 'Sat Feb 17 2018 14:02:13 GMT+0100 (CET)',
125 | salaryFrom: 3640000,
126 | salaryTo: 6160000,
127 | city: 'Vienna'
128 | }, {
129 | id: 6,
130 | slug: 'regular-javascript-engineer',
131 | title: 'Regular JavaScript Engineer',
132 | createdAt: 'Sat Feb 17 2018 13:53:12 GMT+0100 (CET)',
133 | salaryFrom: 4060000,
134 | salaryTo: 5600000,
135 | city: 'Vienna'
136 | }
137 | ];
138 |
139 | beforeAll(() => provider.addInteraction({
140 | state: 'has some job offers',
141 | uponReceiving: 'a request for the job offers',
142 | withRequest: {
143 | method: 'GET',
144 | // endpoint on the back-end for web-ui side
145 | path: '/v1/jobs',
146 | headers: {
147 | 'Accept': Matchers.somethingLike('application/json')
148 | }
149 | },
150 | willRespondWith: {
151 | status: 200,
152 | headers: {
153 | 'Content-Type': Matchers.somethingLike('application/json')
154 | },
155 | body: LIST_BODY
156 | }
157 | }));
158 |
159 | it('returns a job list', async () => {
160 | const response = await fetchJobList();
161 | expect(response).toEqual(LIST_BODY);
162 | });
163 | })
164 |
165 | it('successfully verifies', () => provider.verify())
166 | })
167 |
168 |
169 |
170 |
--------------------------------------------------------------------------------
/packages/backend-for-webui/src/controllers/__tests__/Job.test.js:
--------------------------------------------------------------------------------
1 | const path = require('path');
2 | const { Matchers } = require('@pact-foundation/pact');
3 | const { stringify } = require('querystring');
4 |
5 | const { retrieveJob, retrieveList } = require('../Job');
6 |
7 | // it takes about 4 sec to spin up the mock server
8 | jest.setTimeout(20000);
9 |
10 | const JOB_EXPECTED_BODY = {
11 | id: 1,
12 | slug: 'senior-front-end-developer',
13 | title: 'Senior Front End Developer',
14 | createdAt: 'Sat Feb 10 2018 18:12:40 GMT+0100 (CET)',
15 | salaryFrom: 4340000,
16 | salaryTo: 5880000,
17 | city: 'Vienna'
18 | }
19 |
20 | const FIRST_PAGE_BODY = [
21 | {
22 | id: 1,
23 | slug: 'senior-front-end-developer',
24 | title: 'Senior Front End Developer',
25 | createdAt: 'Sat Feb 10 2018 18:12:40 GMT+0100 (CET)',
26 | salaryFrom: 4340000,
27 | salaryTo: 5880000,
28 | city: 'Vienna'
29 | },{
30 | id: 2,
31 | slug: 'senior-php-developer',
32 | title: 'Senior PHP Developer',
33 | createdAt: 'Sat Feb 17 2018 14:38:50 GMT+0100 (CET)',
34 | salaryFrom: 4900000,
35 | salaryTo: 6580000,
36 | city: 'Vienna'
37 | },{
38 | id: 3,
39 | slug: 'back-end-architect',
40 | title: 'Back End Architect',
41 | createdAt: 'Sat Feb 17 2018 14:22:39 GMT+0100 (CET)',
42 | salaryFrom: 4060000,
43 | salaryTo: 6580000,
44 | city: 'Vienna'
45 | },{
46 | id: 4,
47 | slug: 'ui-developer',
48 | title: 'UI Developer',
49 | createdAt: 'Sat Feb 17 2018 14:07:15 GMT+0100 (CET)',
50 | salaryFrom: 4060000,
51 | salaryTo: 6720000,
52 | city: 'Vienna'
53 | },{
54 | id: 5,
55 | slug: 'principal-javascript-manager',
56 | title: 'Principal JavaScript Manager',
57 | createdAt: 'Sat Feb 17 2018 14:02:13 GMT+0100 (CET)',
58 | salaryFrom: 3640000,
59 | salaryTo: 6160000,
60 | city: 'Vienna'
61 | }
62 | ];
63 |
64 | const LAST_PAGE_BODY = [
65 | {
66 | id: 6,
67 | slug: 'regular-javascript-engineer',
68 | title: 'Regular JavaScript Engineer',
69 | createdAt: 'Sat Feb 17 2018 13:53:12 GMT+0100 (CET)',
70 | salaryFrom: 4060000,
71 | salaryTo: 5600000,
72 | city: 'Vienna'
73 | }
74 | ];
75 |
76 | describe('Job Service', () => {
77 | beforeAll(() => provider.setup());
78 |
79 | afterAll(() => provider.finalize());
80 |
81 | describe('#retrieveJob', () => {
82 |
83 | beforeAll(() => provider.addInteraction({
84 | state: 'has a requested job offer',
85 | uponReceiving: 'a request for an job offer with id 1 and slug senior-front-end-developer',
86 | withRequest: {
87 | method: 'GET',
88 | // endpoint on the job service side
89 | path: '/v1/1-senior-front-end-developer',
90 | headers: {
91 | 'Accept': Matchers.somethingLike('application/json')
92 | }
93 | },
94 | willRespondWith: {
95 | status: 200,
96 | headers: {
97 | 'Content-Type': Matchers.somethingLike('application/json')
98 | },
99 | body: JOB_EXPECTED_BODY
100 | }
101 | }));
102 |
103 | beforeAll(() => provider.addInteraction({
104 | state: 'has no offer matching given criteria',
105 | uponReceiving: 'a request for a job offer',
106 | withRequest: {
107 | method: 'GET',
108 | // endpoint on the job service side
109 | path: '/v1/2-non-existent-offer',
110 | headers: {
111 | 'Accept': Matchers.somethingLike('application/json')
112 | }
113 | },
114 | willRespondWith: {
115 | status: 404,
116 | headers: {
117 | 'Content-Type': Matchers.somethingLike('application/json')
118 | },
119 | body: {
120 | status: 'NOT_FOUND'
121 | }
122 | }
123 | }));
124 |
125 | it('returns a successful body', async () => {
126 | const res = {
127 | status: jest.fn(p => res),
128 | json: jest.fn(body => body),
129 | };
130 | const req = {
131 | params: {
132 | id: '1',
133 | slug: 'senior-front-end-developer',
134 | }
135 | };
136 | await retrieveJob(req, res);
137 | expect(res.json).toBeCalledWith(JOB_EXPECTED_BODY);
138 | });
139 |
140 | it('returns a not found response', async () => {
141 | const res = {
142 | status: jest.fn(p => res),
143 | json: jest.fn(body => body),
144 | };
145 | const req = {
146 | params: {
147 | id: '2',
148 | slug: 'non-existent-offer',
149 | }
150 | };
151 | await retrieveJob(req, res);
152 | expect(res.status).toBeCalledWith(404);
153 | expect(res.json).toBeCalledWith({ status: 'NOT_FOUND'});
154 | });
155 | });
156 |
157 | describe('#retrieveList', () => {
158 | beforeAll(() => provider.addInteraction({
159 | state: 'has 5 job offers',
160 | uponReceiving: 'a request for the first 5 job offers',
161 | withRequest: {
162 | method: 'GET',
163 | // endpoint on the job service side
164 | path: '/v1',
165 | query: stringify({
166 | skip: 0,
167 | amount: 5,
168 | }),
169 | headers: {
170 | 'Accept': Matchers.somethingLike('application/json')
171 | }
172 | },
173 | willRespondWith: {
174 | status: 200,
175 | headers: {
176 | 'Content-Type': Matchers.somethingLike('application/json')
177 | },
178 | body: FIRST_PAGE_BODY
179 | }
180 | }));
181 |
182 | beforeAll(() => provider.addInteraction({
183 | state: 'has 1 more job offer',
184 | uponReceiving: 'a request for 5 job offers skipping first 5',
185 | withRequest: {
186 | method: 'GET',
187 | // endpoint on the job service side
188 | path: '/v1',
189 | query: stringify({
190 | skip: 5,
191 | amount: 5,
192 | }),
193 | headers: {
194 | 'Accept': Matchers.somethingLike('application/json')
195 | }
196 | },
197 | willRespondWith: {
198 | status: 200,
199 | headers: {
200 | 'Content-Type': Matchers.somethingLike('application/json')
201 | },
202 | body: LAST_PAGE_BODY
203 | }
204 | }));
205 |
206 | it('returns the first 5 job offers', async () => {
207 | const res = {
208 | status: jest.fn(p => res),
209 | json: jest.fn(body => body),
210 | };
211 | const req = {
212 | url: 'http://localhost:/3001/v1?skip=0&amount=5'
213 | };
214 | await retrieveList(req, res);
215 | expect(res.json).toBeCalledWith(FIRST_PAGE_BODY);
216 | });
217 |
218 | it('returns 1 more job offer skipping the first 5', async () => {
219 | const res = {
220 | status: jest.fn(p => res),
221 | json: jest.fn(body => body),
222 | };
223 | const req = {
224 | url: 'http://localhost:/3001/v1?skip=5&amount=5'
225 | };
226 | await retrieveList(req, res);
227 | expect(res.json).toBeCalledWith(LAST_PAGE_BODY);
228 | });
229 | });
230 |
231 | it('successfully verifies', () => provider.verify())
232 | })
--------------------------------------------------------------------------------
/packages/job-service/package-lock.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "job-service",
3 | "version": "1.0.0",
4 | "lockfileVersion": 1,
5 | "requires": true,
6 | "dependencies": {
7 | "@pact-foundation/pact-node": {
8 | "version": "6.11.0",
9 | "resolved": "https://registry.npmjs.org/@pact-foundation/pact-node/-/pact-node-6.11.0.tgz",
10 | "integrity": "sha1-gxmUY+A7sLAnWIrfnwZ7xCpVecU=",
11 | "dev": true,
12 | "requires": {
13 | "@pact-foundation/pact-standalone": "3.7.0",
14 | "@types/bunyan": "1.8.4",
15 | "@types/q": "1.0.7",
16 | "bunyan": "1.8.12",
17 | "bunyan-prettystream": "0.1.3",
18 | "caporal": "0.10.0",
19 | "check-types": "7.3.0",
20 | "mkdirp": "0.5.1",
21 | "q": "1.5.1",
22 | "request": "2.83.0",
23 | "traverson": "6.0.3",
24 | "traverson-hal": "6.0.0",
25 | "traverson-promise": "0.0.9",
26 | "underscore": "1.8.3",
27 | "unixify": "1.0.0",
28 | "url-join": "4.0.0"
29 | }
30 | },
31 | "@pact-foundation/pact-standalone": {
32 | "version": "3.7.0",
33 | "resolved": "https://registry.npmjs.org/@pact-foundation/pact-standalone/-/pact-standalone-3.7.0.tgz",
34 | "integrity": "sha1-sUA/xIwCCxL/0VT9FyKUhXxn1LI=",
35 | "dev": true,
36 | "requires": {
37 | "bunyan": "1.8.12",
38 | "bunyan-prettystream": "0.1.3"
39 | }
40 | },
41 | "@types/bunyan": {
42 | "version": "1.8.4",
43 | "resolved": "https://registry.npmjs.org/@types/bunyan/-/bunyan-1.8.4.tgz",
44 | "integrity": "sha512-bxOF3fsm69ezKxdcJ7Oo/PsZMOJ+JIV/QJO2IADfScmR3sLulR88dpSnz6+q+9JJ1kD7dXFFgUrGRSKHLkOX7w==",
45 | "dev": true,
46 | "requires": {
47 | "@types/events": "1.2.0",
48 | "@types/node": "9.4.6"
49 | }
50 | },
51 | "@types/events": {
52 | "version": "1.2.0",
53 | "resolved": "https://registry.npmjs.org/@types/events/-/events-1.2.0.tgz",
54 | "integrity": "sha512-KEIlhXnIutzKwRbQkGWb/I4HFqBuUykAdHgDED6xqwXJfONCjF5VoE0cXEiurh3XauygxzeDzgtXUqvLkxFzzA==",
55 | "dev": true
56 | },
57 | "@types/node": {
58 | "version": "9.4.6",
59 | "resolved": "https://registry.npmjs.org/@types/node/-/node-9.4.6.tgz",
60 | "integrity": "sha512-CTUtLb6WqCCgp6P59QintjHWqzf4VL1uPA27bipLAPxFqrtK1gEYllePzTICGqQ8rYsCbpnsNypXjjDzGAAjEQ==",
61 | "dev": true
62 | },
63 | "@types/q": {
64 | "version": "1.0.7",
65 | "resolved": "https://registry.npmjs.org/@types/q/-/q-1.0.7.tgz",
66 | "integrity": "sha512-0WS7XU7sXzQ7J1nbnMKKYdjrrFoO3YtZYgUzeV8JFXffPnHfvSJQleR70I8BOAsOm14i4dyaAZ3YzqIl1YhkXQ==",
67 | "dev": true
68 | },
69 | "accepts": {
70 | "version": "1.3.5",
71 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.5.tgz",
72 | "integrity": "sha1-63d99gEXI6OxTopywIBcjoZ0a9I=",
73 | "requires": {
74 | "mime-types": "2.1.18",
75 | "negotiator": "0.6.1"
76 | }
77 | },
78 | "ajv": {
79 | "version": "5.5.2",
80 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-5.5.2.tgz",
81 | "integrity": "sha1-c7Xuyj+rZT49P5Qis0GtQiBdyWU=",
82 | "dev": true,
83 | "requires": {
84 | "co": "4.6.0",
85 | "fast-deep-equal": "1.1.0",
86 | "fast-json-stable-stringify": "2.0.0",
87 | "json-schema-traverse": "0.3.1"
88 | }
89 | },
90 | "ansi": {
91 | "version": "0.3.1",
92 | "resolved": "https://registry.npmjs.org/ansi/-/ansi-0.3.1.tgz",
93 | "integrity": "sha1-DELU+xcWDVqa8eSEus4cZpIsGyE=",
94 | "dev": true
95 | },
96 | "ansi-escapes": {
97 | "version": "1.4.0",
98 | "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-1.4.0.tgz",
99 | "integrity": "sha1-06ioOzGapneTZisT52HHkRQiMG4=",
100 | "dev": true
101 | },
102 | "ansi-regex": {
103 | "version": "2.1.1",
104 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz",
105 | "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=",
106 | "dev": true
107 | },
108 | "ansi-styles": {
109 | "version": "2.2.1",
110 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz",
111 | "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=",
112 | "dev": true
113 | },
114 | "are-we-there-yet": {
115 | "version": "1.1.4",
116 | "resolved": "https://registry.npmjs.org/are-we-there-yet/-/are-we-there-yet-1.1.4.tgz",
117 | "integrity": "sha1-u13KOCu5TwXhUZQ3PRb9O6HKEQ0=",
118 | "dev": true,
119 | "requires": {
120 | "delegates": "1.0.0",
121 | "readable-stream": "2.3.5"
122 | }
123 | },
124 | "array-flatten": {
125 | "version": "1.1.1",
126 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
127 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI="
128 | },
129 | "asn1": {
130 | "version": "0.2.3",
131 | "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.3.tgz",
132 | "integrity": "sha1-2sh4dxPJlmhJ/IGAd36+nB3fO4Y=",
133 | "dev": true
134 | },
135 | "assert-plus": {
136 | "version": "1.0.0",
137 | "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz",
138 | "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=",
139 | "dev": true
140 | },
141 | "async": {
142 | "version": "1.0.0",
143 | "resolved": "https://registry.npmjs.org/async/-/async-1.0.0.tgz",
144 | "integrity": "sha1-+PwEyjoTeErenhZBr5hXjPvWR6k=",
145 | "dev": true
146 | },
147 | "asynckit": {
148 | "version": "0.4.0",
149 | "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
150 | "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=",
151 | "dev": true
152 | },
153 | "aws-sign2": {
154 | "version": "0.7.0",
155 | "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz",
156 | "integrity": "sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg=",
157 | "dev": true
158 | },
159 | "aws4": {
160 | "version": "1.6.0",
161 | "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.6.0.tgz",
162 | "integrity": "sha1-g+9cqGCysy5KDe7e6MdxudtXRx4=",
163 | "dev": true
164 | },
165 | "balanced-match": {
166 | "version": "1.0.0",
167 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz",
168 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=",
169 | "dev": true,
170 | "optional": true
171 | },
172 | "bcrypt-pbkdf": {
173 | "version": "1.0.1",
174 | "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.1.tgz",
175 | "integrity": "sha1-Y7xdy2EzG5K8Bf1SiVPDNGKgb40=",
176 | "dev": true,
177 | "optional": true,
178 | "requires": {
179 | "tweetnacl": "0.14.5"
180 | }
181 | },
182 | "bluebird": {
183 | "version": "3.5.1",
184 | "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.5.1.tgz",
185 | "integrity": "sha512-MKiLiV+I1AA596t9w1sQJ8jkiSr5+ZKi0WKrYGUn6d1Fx+Ij4tIj+m2WMQSGczs5jZVxV339chE8iwk6F64wjA==",
186 | "dev": true
187 | },
188 | "body-parser": {
189 | "version": "1.18.2",
190 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.18.2.tgz",
191 | "integrity": "sha1-h2eKGdhLR9hZuDGZvVm84iKxBFQ=",
192 | "requires": {
193 | "bytes": "3.0.0",
194 | "content-type": "1.0.4",
195 | "debug": "2.6.9",
196 | "depd": "1.1.2",
197 | "http-errors": "1.6.2",
198 | "iconv-lite": "0.4.19",
199 | "on-finished": "2.3.0",
200 | "qs": "6.5.1",
201 | "raw-body": "2.3.2",
202 | "type-is": "1.6.16"
203 | }
204 | },
205 | "boom": {
206 | "version": "4.3.1",
207 | "resolved": "https://registry.npmjs.org/boom/-/boom-4.3.1.tgz",
208 | "integrity": "sha1-T4owBctKfjiJ90kDD9JbluAdLjE=",
209 | "dev": true,
210 | "requires": {
211 | "hoek": "4.2.1"
212 | }
213 | },
214 | "brace-expansion": {
215 | "version": "1.1.11",
216 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
217 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
218 | "dev": true,
219 | "optional": true,
220 | "requires": {
221 | "balanced-match": "1.0.0",
222 | "concat-map": "0.0.1"
223 | }
224 | },
225 | "bunyan": {
226 | "version": "1.8.12",
227 | "resolved": "https://registry.npmjs.org/bunyan/-/bunyan-1.8.12.tgz",
228 | "integrity": "sha1-8VDw9nSKvdcq6uhPBEA74u8RN5c=",
229 | "dev": true,
230 | "requires": {
231 | "dtrace-provider": "0.8.6",
232 | "moment": "2.21.0",
233 | "mv": "2.1.1",
234 | "safe-json-stringify": "1.1.0"
235 | }
236 | },
237 | "bunyan-prettystream": {
238 | "version": "0.1.3",
239 | "resolved": "https://registry.npmjs.org/bunyan-prettystream/-/bunyan-prettystream-0.1.3.tgz",
240 | "integrity": "sha1-bDtxMmb2rTIAfHtqsemYokU0nZg=",
241 | "dev": true
242 | },
243 | "bytes": {
244 | "version": "3.0.0",
245 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz",
246 | "integrity": "sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg="
247 | },
248 | "caporal": {
249 | "version": "0.10.0",
250 | "resolved": "https://registry.npmjs.org/caporal/-/caporal-0.10.0.tgz",
251 | "integrity": "sha512-Df9b3SxFqXkdvFdc/i4fNU2wFGYezSKTkENew9txshZhfazE8Ts70OKDmhOD7NRJjz81Xd9Jb+wF9XeqM20WsA==",
252 | "dev": true,
253 | "requires": {
254 | "bluebird": "3.5.1",
255 | "chalk": "1.1.3",
256 | "cli-table2": "0.2.0",
257 | "fast-levenshtein": "2.0.6",
258 | "lodash.camelcase": "4.3.0",
259 | "lodash.kebabcase": "4.1.1",
260 | "lodash.merge": "4.6.1",
261 | "micromist": "1.0.2",
262 | "prettyjson": "1.2.1",
263 | "tabtab": "2.2.2",
264 | "winston": "2.4.0"
265 | }
266 | },
267 | "caseless": {
268 | "version": "0.12.0",
269 | "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz",
270 | "integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=",
271 | "dev": true
272 | },
273 | "chalk": {
274 | "version": "1.1.3",
275 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz",
276 | "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=",
277 | "dev": true,
278 | "requires": {
279 | "ansi-styles": "2.2.1",
280 | "escape-string-regexp": "1.0.5",
281 | "has-ansi": "2.0.0",
282 | "strip-ansi": "3.0.1",
283 | "supports-color": "2.0.0"
284 | }
285 | },
286 | "check-types": {
287 | "version": "7.3.0",
288 | "resolved": "https://registry.npmjs.org/check-types/-/check-types-7.3.0.tgz",
289 | "integrity": "sha1-Ro9XGkQ1wkJI9f0MsOjYfDw0Hn0=",
290 | "dev": true
291 | },
292 | "cli-cursor": {
293 | "version": "1.0.2",
294 | "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-1.0.2.tgz",
295 | "integrity": "sha1-ZNo/fValRBLll5S9Ytw1KV6PKYc=",
296 | "dev": true,
297 | "requires": {
298 | "restore-cursor": "1.0.1"
299 | }
300 | },
301 | "cli-table2": {
302 | "version": "0.2.0",
303 | "resolved": "https://registry.npmjs.org/cli-table2/-/cli-table2-0.2.0.tgz",
304 | "integrity": "sha1-LR738hig54biFFQFYtS9F3/jLZc=",
305 | "dev": true,
306 | "requires": {
307 | "colors": "1.1.2",
308 | "lodash": "3.10.1",
309 | "string-width": "1.0.2"
310 | },
311 | "dependencies": {
312 | "lodash": {
313 | "version": "3.10.1",
314 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-3.10.1.tgz",
315 | "integrity": "sha1-W/Rejkm6QYnhfUgnid/RW9FAt7Y=",
316 | "dev": true
317 | }
318 | }
319 | },
320 | "cli-width": {
321 | "version": "2.2.0",
322 | "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.2.0.tgz",
323 | "integrity": "sha1-/xnt6Kml5XkyQUewwR8PvLq+1jk=",
324 | "dev": true
325 | },
326 | "co": {
327 | "version": "4.6.0",
328 | "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz",
329 | "integrity": "sha1-bqa989hTrlTMuOR7+gvz+QMfsYQ=",
330 | "dev": true
331 | },
332 | "code-point-at": {
333 | "version": "1.1.0",
334 | "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz",
335 | "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c=",
336 | "dev": true
337 | },
338 | "colors": {
339 | "version": "1.1.2",
340 | "resolved": "https://registry.npmjs.org/colors/-/colors-1.1.2.tgz",
341 | "integrity": "sha1-FopHAXVran9RoSzgyXv6KMCE7WM=",
342 | "dev": true
343 | },
344 | "combined-stream": {
345 | "version": "1.0.6",
346 | "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.6.tgz",
347 | "integrity": "sha1-cj599ugBrFYTETp+RFqbactjKBg=",
348 | "dev": true,
349 | "requires": {
350 | "delayed-stream": "1.0.0"
351 | }
352 | },
353 | "component-emitter": {
354 | "version": "1.2.1",
355 | "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.2.1.tgz",
356 | "integrity": "sha1-E3kY1teCg/ffemt8WmPhQOaUJeY=",
357 | "dev": true
358 | },
359 | "concat-map": {
360 | "version": "0.0.1",
361 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
362 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=",
363 | "dev": true,
364 | "optional": true
365 | },
366 | "concat-stream": {
367 | "version": "1.6.1",
368 | "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.1.tgz",
369 | "integrity": "sha512-gslSSJx03QKa59cIKqeJO9HQ/WZMotvYJCuaUULrLpjj8oG40kV2Z+gz82pVxlTkOADi4PJxQPPfhl1ELYrrXw==",
370 | "dev": true,
371 | "requires": {
372 | "inherits": "2.0.3",
373 | "readable-stream": "2.3.5",
374 | "typedarray": "0.0.6"
375 | }
376 | },
377 | "content-disposition": {
378 | "version": "0.5.2",
379 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.2.tgz",
380 | "integrity": "sha1-DPaLud318r55YcOoUXjLhdunjLQ="
381 | },
382 | "content-type": {
383 | "version": "1.0.4",
384 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz",
385 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA=="
386 | },
387 | "cookie": {
388 | "version": "0.3.1",
389 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.3.1.tgz",
390 | "integrity": "sha1-5+Ch+e9DtMi6klxcWpboBtFoc7s="
391 | },
392 | "cookie-signature": {
393 | "version": "1.0.6",
394 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
395 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw="
396 | },
397 | "cookiejar": {
398 | "version": "2.1.1",
399 | "resolved": "https://registry.npmjs.org/cookiejar/-/cookiejar-2.1.1.tgz",
400 | "integrity": "sha1-Qa1XsbVVlR7BcUEqgZQrHoIA00o=",
401 | "dev": true
402 | },
403 | "core-util-is": {
404 | "version": "1.0.2",
405 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz",
406 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=",
407 | "dev": true
408 | },
409 | "cross-env": {
410 | "version": "5.1.3",
411 | "resolved": "https://registry.npmjs.org/cross-env/-/cross-env-5.1.3.tgz",
412 | "integrity": "sha512-UOokgwvDzCT0mqRSLEkJzUhYXB1vK3E5UgDrD41QiXsm9UetcW2rCGHYz/O3p873lMJ1VZbFCF9Izkwh7nYR5A==",
413 | "dev": true,
414 | "requires": {
415 | "cross-spawn": "5.1.0",
416 | "is-windows": "1.0.2"
417 | }
418 | },
419 | "cross-spawn": {
420 | "version": "5.1.0",
421 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-5.1.0.tgz",
422 | "integrity": "sha1-6L0O/uWPz/b4+UUQoKVUu/ojVEk=",
423 | "dev": true,
424 | "requires": {
425 | "lru-cache": "4.1.1",
426 | "shebang-command": "1.2.0",
427 | "which": "1.3.0"
428 | }
429 | },
430 | "cryptiles": {
431 | "version": "3.1.2",
432 | "resolved": "https://registry.npmjs.org/cryptiles/-/cryptiles-3.1.2.tgz",
433 | "integrity": "sha1-qJ+7Ig9c4l7FboxKqKT9e1sNKf4=",
434 | "dev": true,
435 | "requires": {
436 | "boom": "5.2.0"
437 | },
438 | "dependencies": {
439 | "boom": {
440 | "version": "5.2.0",
441 | "resolved": "https://registry.npmjs.org/boom/-/boom-5.2.0.tgz",
442 | "integrity": "sha512-Z5BTk6ZRe4tXXQlkqftmsAUANpXmuwlsF5Oov8ThoMbQRzdGTA1ngYRW160GexgOgjsFOKJz0LYhoNi+2AMBUw==",
443 | "dev": true,
444 | "requires": {
445 | "hoek": "4.2.1"
446 | }
447 | }
448 | }
449 | },
450 | "cycle": {
451 | "version": "1.0.3",
452 | "resolved": "https://registry.npmjs.org/cycle/-/cycle-1.0.3.tgz",
453 | "integrity": "sha1-IegLK+hYD5i0aPN5QwZisEbDStI=",
454 | "dev": true
455 | },
456 | "dashdash": {
457 | "version": "1.14.1",
458 | "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz",
459 | "integrity": "sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=",
460 | "dev": true,
461 | "requires": {
462 | "assert-plus": "1.0.0"
463 | }
464 | },
465 | "debug": {
466 | "version": "2.6.9",
467 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
468 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
469 | "requires": {
470 | "ms": "2.0.0"
471 | }
472 | },
473 | "delayed-stream": {
474 | "version": "1.0.0",
475 | "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
476 | "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=",
477 | "dev": true
478 | },
479 | "delegates": {
480 | "version": "1.0.0",
481 | "resolved": "https://registry.npmjs.org/delegates/-/delegates-1.0.0.tgz",
482 | "integrity": "sha1-hMbhWbgZBP3KWaDvRM2HDTElD5o=",
483 | "dev": true
484 | },
485 | "depd": {
486 | "version": "1.1.2",
487 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz",
488 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak="
489 | },
490 | "destroy": {
491 | "version": "1.0.4",
492 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz",
493 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA="
494 | },
495 | "dtrace-provider": {
496 | "version": "0.8.6",
497 | "resolved": "https://registry.npmjs.org/dtrace-provider/-/dtrace-provider-0.8.6.tgz",
498 | "integrity": "sha1-QooiOv4DQl0s1tY0f99AxmkDVj0=",
499 | "dev": true,
500 | "optional": true,
501 | "requires": {
502 | "nan": "2.9.2"
503 | }
504 | },
505 | "ecc-jsbn": {
506 | "version": "0.1.1",
507 | "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.1.tgz",
508 | "integrity": "sha1-D8c6ntXw1Tw4GTOYUj735UN3dQU=",
509 | "dev": true,
510 | "optional": true,
511 | "requires": {
512 | "jsbn": "0.1.1"
513 | }
514 | },
515 | "ee-first": {
516 | "version": "1.1.1",
517 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
518 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0="
519 | },
520 | "encodeurl": {
521 | "version": "1.0.2",
522 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
523 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k="
524 | },
525 | "escape-html": {
526 | "version": "1.0.3",
527 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
528 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg="
529 | },
530 | "escape-string-regexp": {
531 | "version": "1.0.5",
532 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz",
533 | "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=",
534 | "dev": true
535 | },
536 | "etag": {
537 | "version": "1.8.1",
538 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
539 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc="
540 | },
541 | "exit-hook": {
542 | "version": "1.1.1",
543 | "resolved": "https://registry.npmjs.org/exit-hook/-/exit-hook-1.1.1.tgz",
544 | "integrity": "sha1-8FyiM7SMBdVP/wd2XfhQfpXAL/g=",
545 | "dev": true
546 | },
547 | "express": {
548 | "version": "4.16.2",
549 | "resolved": "https://registry.npmjs.org/express/-/express-4.16.2.tgz",
550 | "integrity": "sha1-41xt/i1kt9ygpc1PIXgb4ymeB2w=",
551 | "requires": {
552 | "accepts": "1.3.5",
553 | "array-flatten": "1.1.1",
554 | "body-parser": "1.18.2",
555 | "content-disposition": "0.5.2",
556 | "content-type": "1.0.4",
557 | "cookie": "0.3.1",
558 | "cookie-signature": "1.0.6",
559 | "debug": "2.6.9",
560 | "depd": "1.1.2",
561 | "encodeurl": "1.0.2",
562 | "escape-html": "1.0.3",
563 | "etag": "1.8.1",
564 | "finalhandler": "1.1.0",
565 | "fresh": "0.5.2",
566 | "merge-descriptors": "1.0.1",
567 | "methods": "1.1.2",
568 | "on-finished": "2.3.0",
569 | "parseurl": "1.3.2",
570 | "path-to-regexp": "0.1.7",
571 | "proxy-addr": "2.0.3",
572 | "qs": "6.5.1",
573 | "range-parser": "1.2.0",
574 | "safe-buffer": "5.1.1",
575 | "send": "0.16.1",
576 | "serve-static": "1.13.1",
577 | "setprototypeof": "1.1.0",
578 | "statuses": "1.3.1",
579 | "type-is": "1.6.16",
580 | "utils-merge": "1.0.1",
581 | "vary": "1.1.2"
582 | }
583 | },
584 | "extend": {
585 | "version": "3.0.1",
586 | "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.1.tgz",
587 | "integrity": "sha1-p1Xqe8Gt/MWjHOfnYtuq3F5jZEQ=",
588 | "dev": true
589 | },
590 | "external-editor": {
591 | "version": "1.1.1",
592 | "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-1.1.1.tgz",
593 | "integrity": "sha1-Etew24UPf/fnCBuvQAVwAGDEYAs=",
594 | "dev": true,
595 | "requires": {
596 | "extend": "3.0.1",
597 | "spawn-sync": "1.0.15",
598 | "tmp": "0.0.29"
599 | }
600 | },
601 | "extsprintf": {
602 | "version": "1.3.0",
603 | "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz",
604 | "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=",
605 | "dev": true
606 | },
607 | "eyes": {
608 | "version": "0.1.8",
609 | "resolved": "https://registry.npmjs.org/eyes/-/eyes-0.1.8.tgz",
610 | "integrity": "sha1-Ys8SAjTGg3hdkCNIqADvPgzCC8A=",
611 | "dev": true
612 | },
613 | "fast-deep-equal": {
614 | "version": "1.1.0",
615 | "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-1.1.0.tgz",
616 | "integrity": "sha1-wFNHeBfIa1HaqFPIHgWbcz0CNhQ=",
617 | "dev": true
618 | },
619 | "fast-json-stable-stringify": {
620 | "version": "2.0.0",
621 | "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.0.0.tgz",
622 | "integrity": "sha1-1RQsDK7msRifh9OnYREGT4bIu/I=",
623 | "dev": true
624 | },
625 | "fast-levenshtein": {
626 | "version": "2.0.6",
627 | "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz",
628 | "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=",
629 | "dev": true
630 | },
631 | "figures": {
632 | "version": "1.7.0",
633 | "resolved": "https://registry.npmjs.org/figures/-/figures-1.7.0.tgz",
634 | "integrity": "sha1-y+Hjr/zxzUS4DK3+0o3Hk6lwHS4=",
635 | "dev": true,
636 | "requires": {
637 | "escape-string-regexp": "1.0.5",
638 | "object-assign": "4.1.1"
639 | }
640 | },
641 | "finalhandler": {
642 | "version": "1.1.0",
643 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.0.tgz",
644 | "integrity": "sha1-zgtoVbRYU+eRsvzGgARtiCU91/U=",
645 | "requires": {
646 | "debug": "2.6.9",
647 | "encodeurl": "1.0.2",
648 | "escape-html": "1.0.3",
649 | "on-finished": "2.3.0",
650 | "parseurl": "1.3.2",
651 | "statuses": "1.3.1",
652 | "unpipe": "1.0.0"
653 | }
654 | },
655 | "forever-agent": {
656 | "version": "0.6.1",
657 | "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz",
658 | "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=",
659 | "dev": true
660 | },
661 | "form-data": {
662 | "version": "2.3.2",
663 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.2.tgz",
664 | "integrity": "sha1-SXBJi+YEwgwAXU9cI67NIda0kJk=",
665 | "dev": true,
666 | "requires": {
667 | "asynckit": "0.4.0",
668 | "combined-stream": "1.0.6",
669 | "mime-types": "2.1.18"
670 | }
671 | },
672 | "formidable": {
673 | "version": "1.1.1",
674 | "resolved": "https://registry.npmjs.org/formidable/-/formidable-1.1.1.tgz",
675 | "integrity": "sha1-lriIb3w8NQi5Mta9cMTTqI818ak=",
676 | "dev": true
677 | },
678 | "forwarded": {
679 | "version": "0.1.2",
680 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz",
681 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ="
682 | },
683 | "fresh": {
684 | "version": "0.5.2",
685 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
686 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac="
687 | },
688 | "gauge": {
689 | "version": "1.2.7",
690 | "resolved": "https://registry.npmjs.org/gauge/-/gauge-1.2.7.tgz",
691 | "integrity": "sha1-6c7FSD09TuDvRLYKfZnkk14TbZM=",
692 | "dev": true,
693 | "requires": {
694 | "ansi": "0.3.1",
695 | "has-unicode": "2.0.1",
696 | "lodash.pad": "4.5.1",
697 | "lodash.padend": "4.6.1",
698 | "lodash.padstart": "4.6.1"
699 | }
700 | },
701 | "getpass": {
702 | "version": "0.1.7",
703 | "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz",
704 | "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=",
705 | "dev": true,
706 | "requires": {
707 | "assert-plus": "1.0.0"
708 | }
709 | },
710 | "glob": {
711 | "version": "6.0.4",
712 | "resolved": "https://registry.npmjs.org/glob/-/glob-6.0.4.tgz",
713 | "integrity": "sha1-DwiGD2oVUSey+t1PnOJLGqtuTSI=",
714 | "dev": true,
715 | "optional": true,
716 | "requires": {
717 | "inflight": "1.0.6",
718 | "inherits": "2.0.3",
719 | "minimatch": "3.0.4",
720 | "once": "1.4.0",
721 | "path-is-absolute": "1.0.1"
722 | }
723 | },
724 | "halfred": {
725 | "version": "1.0.0",
726 | "resolved": "https://registry.npmjs.org/halfred/-/halfred-1.0.0.tgz",
727 | "integrity": "sha1-dC1FtNZkvGOkPWAWgC5iQUyt2iI=",
728 | "dev": true
729 | },
730 | "har-schema": {
731 | "version": "2.0.0",
732 | "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz",
733 | "integrity": "sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI=",
734 | "dev": true
735 | },
736 | "har-validator": {
737 | "version": "5.0.3",
738 | "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.0.3.tgz",
739 | "integrity": "sha1-ukAsJmGU8VlW7xXg/PJCmT9qff0=",
740 | "dev": true,
741 | "requires": {
742 | "ajv": "5.5.2",
743 | "har-schema": "2.0.0"
744 | }
745 | },
746 | "has-ansi": {
747 | "version": "2.0.0",
748 | "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz",
749 | "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=",
750 | "dev": true,
751 | "requires": {
752 | "ansi-regex": "2.1.1"
753 | }
754 | },
755 | "has-unicode": {
756 | "version": "2.0.1",
757 | "resolved": "https://registry.npmjs.org/has-unicode/-/has-unicode-2.0.1.tgz",
758 | "integrity": "sha1-4Ob+aijPUROIVeCG0Wkedx3iqLk=",
759 | "dev": true
760 | },
761 | "hawk": {
762 | "version": "6.0.2",
763 | "resolved": "https://registry.npmjs.org/hawk/-/hawk-6.0.2.tgz",
764 | "integrity": "sha512-miowhl2+U7Qle4vdLqDdPt9m09K6yZhkLDTWGoUiUzrQCn+mHHSmfJgAyGaLRZbPmTqfFFjRV1QWCW0VWUJBbQ==",
765 | "dev": true,
766 | "requires": {
767 | "boom": "4.3.1",
768 | "cryptiles": "3.1.2",
769 | "hoek": "4.2.1",
770 | "sntp": "2.1.0"
771 | }
772 | },
773 | "hoek": {
774 | "version": "4.2.1",
775 | "resolved": "https://registry.npmjs.org/hoek/-/hoek-4.2.1.tgz",
776 | "integrity": "sha512-QLg82fGkfnJ/4iy1xZ81/9SIJiq1NGFUMGs6ParyjBZr6jW2Ufj/snDqTHixNlHdPNwN2RLVD0Pi3igeK9+JfA==",
777 | "dev": true
778 | },
779 | "http-errors": {
780 | "version": "1.6.2",
781 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.2.tgz",
782 | "integrity": "sha1-CgAsyFcHGSp+eUbO7cERVfYOxzY=",
783 | "requires": {
784 | "depd": "1.1.1",
785 | "inherits": "2.0.3",
786 | "setprototypeof": "1.0.3",
787 | "statuses": "1.3.1"
788 | },
789 | "dependencies": {
790 | "depd": {
791 | "version": "1.1.1",
792 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.1.tgz",
793 | "integrity": "sha1-V4O04cRZ8G+lyif5kfPQbnoxA1k="
794 | },
795 | "setprototypeof": {
796 | "version": "1.0.3",
797 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.0.3.tgz",
798 | "integrity": "sha1-ZlZ+NwQ+608E2RvWWMDL77VbjgQ="
799 | }
800 | }
801 | },
802 | "http-signature": {
803 | "version": "1.2.0",
804 | "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz",
805 | "integrity": "sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=",
806 | "dev": true,
807 | "requires": {
808 | "assert-plus": "1.0.0",
809 | "jsprim": "1.4.1",
810 | "sshpk": "1.13.1"
811 | }
812 | },
813 | "iconv-lite": {
814 | "version": "0.4.19",
815 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.19.tgz",
816 | "integrity": "sha512-oTZqweIP51xaGPI4uPa56/Pri/480R+mo7SeU+YETByQNhDG55ycFyNLIgta9vXhILrxXDmF7ZGhqZIcuN0gJQ=="
817 | },
818 | "inflight": {
819 | "version": "1.0.6",
820 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz",
821 | "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=",
822 | "dev": true,
823 | "optional": true,
824 | "requires": {
825 | "once": "1.4.0",
826 | "wrappy": "1.0.2"
827 | }
828 | },
829 | "inherits": {
830 | "version": "2.0.3",
831 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
832 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4="
833 | },
834 | "inquirer": {
835 | "version": "1.2.3",
836 | "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-1.2.3.tgz",
837 | "integrity": "sha1-TexvMvN+97sLLtPx0aXD9UUHSRg=",
838 | "dev": true,
839 | "requires": {
840 | "ansi-escapes": "1.4.0",
841 | "chalk": "1.1.3",
842 | "cli-cursor": "1.0.2",
843 | "cli-width": "2.2.0",
844 | "external-editor": "1.1.1",
845 | "figures": "1.7.0",
846 | "lodash": "4.17.5",
847 | "mute-stream": "0.0.6",
848 | "pinkie-promise": "2.0.1",
849 | "run-async": "2.3.0",
850 | "rx": "4.1.0",
851 | "string-width": "1.0.2",
852 | "strip-ansi": "3.0.1",
853 | "through": "2.3.8"
854 | }
855 | },
856 | "ipaddr.js": {
857 | "version": "1.6.0",
858 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.6.0.tgz",
859 | "integrity": "sha1-4/o1e3c9phnybpXwSdBVxyeW+Gs="
860 | },
861 | "is-fullwidth-code-point": {
862 | "version": "1.0.0",
863 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz",
864 | "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=",
865 | "dev": true,
866 | "requires": {
867 | "number-is-nan": "1.0.1"
868 | }
869 | },
870 | "is-promise": {
871 | "version": "2.1.0",
872 | "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-2.1.0.tgz",
873 | "integrity": "sha1-eaKp7OfwlugPNtKy87wWwf9L8/o=",
874 | "dev": true
875 | },
876 | "is-typedarray": {
877 | "version": "1.0.0",
878 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz",
879 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=",
880 | "dev": true
881 | },
882 | "is-windows": {
883 | "version": "1.0.2",
884 | "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz",
885 | "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==",
886 | "dev": true
887 | },
888 | "isarray": {
889 | "version": "1.0.0",
890 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz",
891 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=",
892 | "dev": true
893 | },
894 | "isexe": {
895 | "version": "2.0.0",
896 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz",
897 | "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=",
898 | "dev": true
899 | },
900 | "isstream": {
901 | "version": "0.1.2",
902 | "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz",
903 | "integrity": "sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo=",
904 | "dev": true
905 | },
906 | "jsbn": {
907 | "version": "0.1.1",
908 | "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz",
909 | "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=",
910 | "dev": true,
911 | "optional": true
912 | },
913 | "json-schema": {
914 | "version": "0.2.3",
915 | "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.2.3.tgz",
916 | "integrity": "sha1-tIDIkuWaLwWVTOcnvT8qTogvnhM=",
917 | "dev": true
918 | },
919 | "json-schema-traverse": {
920 | "version": "0.3.1",
921 | "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.3.1.tgz",
922 | "integrity": "sha1-NJptRMU6Ud6JtAgFxdXlm0F9M0A=",
923 | "dev": true
924 | },
925 | "json-stringify-safe": {
926 | "version": "5.0.1",
927 | "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz",
928 | "integrity": "sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=",
929 | "dev": true
930 | },
931 | "jsonpath-plus": {
932 | "version": "0.16.0",
933 | "resolved": "https://registry.npmjs.org/jsonpath-plus/-/jsonpath-plus-0.16.0.tgz",
934 | "integrity": "sha1-/kQbI/A+xpeaVgNROYjNPtt9tdw=",
935 | "dev": true
936 | },
937 | "jsprim": {
938 | "version": "1.4.1",
939 | "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.1.tgz",
940 | "integrity": "sha1-MT5mvB5cwG5Di8G3SZwuXFastqI=",
941 | "dev": true,
942 | "requires": {
943 | "assert-plus": "1.0.0",
944 | "extsprintf": "1.3.0",
945 | "json-schema": "0.2.3",
946 | "verror": "1.10.0"
947 | }
948 | },
949 | "lodash": {
950 | "version": "4.17.5",
951 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.5.tgz",
952 | "integrity": "sha512-svL3uiZf1RwhH+cWrfZn3A4+U58wbP0tGVTLQPbjplZxZ8ROD9VLuNgsRniTlLe7OlSqR79RUehXgpBW/s0IQw=="
953 | },
954 | "lodash.camelcase": {
955 | "version": "4.3.0",
956 | "resolved": "https://registry.npmjs.org/lodash.camelcase/-/lodash.camelcase-4.3.0.tgz",
957 | "integrity": "sha1-soqmKIorn8ZRA1x3EfZathkDMaY=",
958 | "dev": true
959 | },
960 | "lodash.difference": {
961 | "version": "4.5.0",
962 | "resolved": "https://registry.npmjs.org/lodash.difference/-/lodash.difference-4.5.0.tgz",
963 | "integrity": "sha1-nMtOUF1Ia5FlE0V3KIWi3yf9AXw=",
964 | "dev": true
965 | },
966 | "lodash.kebabcase": {
967 | "version": "4.1.1",
968 | "resolved": "https://registry.npmjs.org/lodash.kebabcase/-/lodash.kebabcase-4.1.1.tgz",
969 | "integrity": "sha1-hImxyw0p/4gZXM7KRI/21swpXDY=",
970 | "dev": true
971 | },
972 | "lodash.merge": {
973 | "version": "4.6.1",
974 | "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.1.tgz",
975 | "integrity": "sha512-AOYza4+Hf5z1/0Hztxpm2/xiPZgi/cjMqdnKTUWTBSKchJlxXXuUSxCCl8rJlf4g6yww/j6mA8nC8Hw/EZWxKQ==",
976 | "dev": true
977 | },
978 | "lodash.pad": {
979 | "version": "4.5.1",
980 | "resolved": "https://registry.npmjs.org/lodash.pad/-/lodash.pad-4.5.1.tgz",
981 | "integrity": "sha1-QzCUmoM6fI2iLMIPaibE1Z3runA=",
982 | "dev": true
983 | },
984 | "lodash.padend": {
985 | "version": "4.6.1",
986 | "resolved": "https://registry.npmjs.org/lodash.padend/-/lodash.padend-4.6.1.tgz",
987 | "integrity": "sha1-U8y6BH0G4VjTEfRdpiX05J5vFm4=",
988 | "dev": true
989 | },
990 | "lodash.padstart": {
991 | "version": "4.6.1",
992 | "resolved": "https://registry.npmjs.org/lodash.padstart/-/lodash.padstart-4.6.1.tgz",
993 | "integrity": "sha1-0uPuv/DZ05rVD1y9G1KnvOa7YRs=",
994 | "dev": true
995 | },
996 | "lodash.uniq": {
997 | "version": "4.5.0",
998 | "resolved": "https://registry.npmjs.org/lodash.uniq/-/lodash.uniq-4.5.0.tgz",
999 | "integrity": "sha1-0CJTc662Uq3BvILklFM5qEJ1R3M=",
1000 | "dev": true
1001 | },
1002 | "lru-cache": {
1003 | "version": "4.1.1",
1004 | "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-4.1.1.tgz",
1005 | "integrity": "sha512-q4spe4KTfsAS1SUHLO0wz8Qiyf1+vMIAgpRYioFYDMNqKfHQbg+AVDH3i4fvpl71/P1L0dBl+fQi+P37UYf0ew==",
1006 | "dev": true,
1007 | "requires": {
1008 | "pseudomap": "1.0.2",
1009 | "yallist": "2.1.2"
1010 | }
1011 | },
1012 | "media-typer": {
1013 | "version": "0.3.0",
1014 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
1015 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g="
1016 | },
1017 | "merge-descriptors": {
1018 | "version": "1.0.1",
1019 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
1020 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E="
1021 | },
1022 | "methods": {
1023 | "version": "1.1.2",
1024 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
1025 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4="
1026 | },
1027 | "microee": {
1028 | "version": "0.0.6",
1029 | "resolved": "https://registry.npmjs.org/microee/-/microee-0.0.6.tgz",
1030 | "integrity": "sha1-oSvbAQNoHosSapsHHrpMRnx4//4=",
1031 | "dev": true
1032 | },
1033 | "micromist": {
1034 | "version": "1.0.2",
1035 | "resolved": "https://registry.npmjs.org/micromist/-/micromist-1.0.2.tgz",
1036 | "integrity": "sha1-QfhJSaBMMM3GCjlNDLBqqgi4Y2Q=",
1037 | "dev": true,
1038 | "requires": {
1039 | "lodash.camelcase": "4.3.0"
1040 | }
1041 | },
1042 | "mime": {
1043 | "version": "1.4.1",
1044 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.4.1.tgz",
1045 | "integrity": "sha512-KI1+qOZu5DcW6wayYHSzR/tXKCDC5Om4s1z2QJjDULzLcmf3DvzS7oluY4HCTrc+9FiKmWUgeNLg7W3uIQvxtQ=="
1046 | },
1047 | "mime-db": {
1048 | "version": "1.33.0",
1049 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.33.0.tgz",
1050 | "integrity": "sha512-BHJ/EKruNIqJf/QahvxwQZXKygOQ256myeN/Ew+THcAa5q+PjyTTMMeNQC4DZw5AwfvelsUrA6B67NKMqXDbzQ=="
1051 | },
1052 | "mime-types": {
1053 | "version": "2.1.18",
1054 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.18.tgz",
1055 | "integrity": "sha512-lc/aahn+t4/SWV/qcmumYjymLsWfN3ELhpmVuUFjgsORruuZPVSwAQryq+HHGvO/SI2KVX26bx+En+zhM8g8hQ==",
1056 | "requires": {
1057 | "mime-db": "1.33.0"
1058 | }
1059 | },
1060 | "minilog": {
1061 | "version": "3.1.0",
1062 | "resolved": "https://registry.npmjs.org/minilog/-/minilog-3.1.0.tgz",
1063 | "integrity": "sha1-0tDxiHyjY9Gs8OqG1cTfKTs/tnU=",
1064 | "dev": true,
1065 | "requires": {
1066 | "microee": "0.0.6"
1067 | }
1068 | },
1069 | "minimatch": {
1070 | "version": "3.0.4",
1071 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz",
1072 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
1073 | "dev": true,
1074 | "optional": true,
1075 | "requires": {
1076 | "brace-expansion": "1.1.11"
1077 | }
1078 | },
1079 | "minimist": {
1080 | "version": "0.0.8",
1081 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz",
1082 | "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=",
1083 | "dev": true
1084 | },
1085 | "mkdirp": {
1086 | "version": "0.5.1",
1087 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz",
1088 | "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=",
1089 | "dev": true,
1090 | "requires": {
1091 | "minimist": "0.0.8"
1092 | }
1093 | },
1094 | "moment": {
1095 | "version": "2.21.0",
1096 | "resolved": "https://registry.npmjs.org/moment/-/moment-2.21.0.tgz",
1097 | "integrity": "sha512-TCZ36BjURTeFTM/CwRcViQlfkMvL1/vFISuNLO5GkcVm1+QHfbSiNqZuWeMFjj1/3+uAjXswgRk30j1kkLYJBQ==",
1098 | "dev": true,
1099 | "optional": true
1100 | },
1101 | "ms": {
1102 | "version": "2.0.0",
1103 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
1104 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
1105 | },
1106 | "mute-stream": {
1107 | "version": "0.0.6",
1108 | "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.6.tgz",
1109 | "integrity": "sha1-SJYrGeFp/R38JAs/HnMXYnu8R9s=",
1110 | "dev": true
1111 | },
1112 | "mv": {
1113 | "version": "2.1.1",
1114 | "resolved": "https://registry.npmjs.org/mv/-/mv-2.1.1.tgz",
1115 | "integrity": "sha1-rmzg1vbV4KT32JN5jQPB6pVZtqI=",
1116 | "dev": true,
1117 | "optional": true,
1118 | "requires": {
1119 | "mkdirp": "0.5.1",
1120 | "ncp": "2.0.0",
1121 | "rimraf": "2.4.5"
1122 | }
1123 | },
1124 | "nan": {
1125 | "version": "2.9.2",
1126 | "resolved": "https://registry.npmjs.org/nan/-/nan-2.9.2.tgz",
1127 | "integrity": "sha512-ltW65co7f3PQWBDbqVvaU1WtFJUsNW7sWWm4HINhbMQIyVyzIeyZ8toX5TC5eeooE6piZoaEh4cZkueSKG3KYw==",
1128 | "dev": true,
1129 | "optional": true
1130 | },
1131 | "ncp": {
1132 | "version": "2.0.0",
1133 | "resolved": "https://registry.npmjs.org/ncp/-/ncp-2.0.0.tgz",
1134 | "integrity": "sha1-GVoh1sRuNh0vsSgbo4uR6d9727M=",
1135 | "dev": true,
1136 | "optional": true
1137 | },
1138 | "negotiator": {
1139 | "version": "0.6.1",
1140 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.1.tgz",
1141 | "integrity": "sha1-KzJxhOiZIQEXeyhWP7XnECrNDKk="
1142 | },
1143 | "normalize-path": {
1144 | "version": "2.1.1",
1145 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-2.1.1.tgz",
1146 | "integrity": "sha1-GrKLVW4Zg2Oowab35vogE3/mrtk=",
1147 | "dev": true,
1148 | "requires": {
1149 | "remove-trailing-separator": "1.1.0"
1150 | }
1151 | },
1152 | "npmlog": {
1153 | "version": "2.0.4",
1154 | "resolved": "https://registry.npmjs.org/npmlog/-/npmlog-2.0.4.tgz",
1155 | "integrity": "sha1-mLUlMPJRTKkNCexbIsiEZyI3VpI=",
1156 | "dev": true,
1157 | "requires": {
1158 | "ansi": "0.3.1",
1159 | "are-we-there-yet": "1.1.4",
1160 | "gauge": "1.2.7"
1161 | }
1162 | },
1163 | "number-is-nan": {
1164 | "version": "1.0.1",
1165 | "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz",
1166 | "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=",
1167 | "dev": true
1168 | },
1169 | "oauth-sign": {
1170 | "version": "0.8.2",
1171 | "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.8.2.tgz",
1172 | "integrity": "sha1-Rqarfwrq2N6unsBWV4C31O/rnUM=",
1173 | "dev": true
1174 | },
1175 | "object-assign": {
1176 | "version": "4.1.1",
1177 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
1178 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=",
1179 | "dev": true
1180 | },
1181 | "on-finished": {
1182 | "version": "2.3.0",
1183 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz",
1184 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=",
1185 | "requires": {
1186 | "ee-first": "1.1.1"
1187 | }
1188 | },
1189 | "once": {
1190 | "version": "1.4.0",
1191 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
1192 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=",
1193 | "dev": true,
1194 | "requires": {
1195 | "wrappy": "1.0.2"
1196 | }
1197 | },
1198 | "onetime": {
1199 | "version": "1.1.0",
1200 | "resolved": "https://registry.npmjs.org/onetime/-/onetime-1.1.0.tgz",
1201 | "integrity": "sha1-ofeDj4MUxRbwXs78vEzP4EtO14k=",
1202 | "dev": true
1203 | },
1204 | "os-shim": {
1205 | "version": "0.1.3",
1206 | "resolved": "https://registry.npmjs.org/os-shim/-/os-shim-0.1.3.tgz",
1207 | "integrity": "sha1-a2LDeRz3kJ6jXtRuF2WLtBfLORc=",
1208 | "dev": true
1209 | },
1210 | "os-tmpdir": {
1211 | "version": "1.0.2",
1212 | "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz",
1213 | "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=",
1214 | "dev": true
1215 | },
1216 | "parseurl": {
1217 | "version": "1.3.2",
1218 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.2.tgz",
1219 | "integrity": "sha1-/CidTtiZMRlGDBViUyYs3I3mW/M="
1220 | },
1221 | "path-is-absolute": {
1222 | "version": "1.0.1",
1223 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz",
1224 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=",
1225 | "dev": true,
1226 | "optional": true
1227 | },
1228 | "path-to-regexp": {
1229 | "version": "0.1.7",
1230 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
1231 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w="
1232 | },
1233 | "performance-now": {
1234 | "version": "2.1.0",
1235 | "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz",
1236 | "integrity": "sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=",
1237 | "dev": true
1238 | },
1239 | "pinkie": {
1240 | "version": "2.0.4",
1241 | "resolved": "https://registry.npmjs.org/pinkie/-/pinkie-2.0.4.tgz",
1242 | "integrity": "sha1-clVrgM+g1IqXToDnckjoDtT3+HA=",
1243 | "dev": true
1244 | },
1245 | "pinkie-promise": {
1246 | "version": "2.0.1",
1247 | "resolved": "https://registry.npmjs.org/pinkie-promise/-/pinkie-promise-2.0.1.tgz",
1248 | "integrity": "sha1-ITXW36ejWMBprJsXh3YogihFD/o=",
1249 | "dev": true,
1250 | "requires": {
1251 | "pinkie": "2.0.4"
1252 | }
1253 | },
1254 | "prettyjson": {
1255 | "version": "1.2.1",
1256 | "resolved": "https://registry.npmjs.org/prettyjson/-/prettyjson-1.2.1.tgz",
1257 | "integrity": "sha1-/P+rQdGcq0365eV15kJGYZsS0ok=",
1258 | "dev": true,
1259 | "requires": {
1260 | "colors": "1.1.2",
1261 | "minimist": "1.2.0"
1262 | },
1263 | "dependencies": {
1264 | "minimist": {
1265 | "version": "1.2.0",
1266 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz",
1267 | "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ=",
1268 | "dev": true
1269 | }
1270 | }
1271 | },
1272 | "process-nextick-args": {
1273 | "version": "2.0.0",
1274 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.0.tgz",
1275 | "integrity": "sha512-MtEC1TqN0EU5nephaJ4rAtThHtC86dNN9qCuEhtshvpVBkAW5ZO7BASN9REnF9eoXGcRub+pFuKEpOHE+HbEMw==",
1276 | "dev": true
1277 | },
1278 | "proxy-addr": {
1279 | "version": "2.0.3",
1280 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.3.tgz",
1281 | "integrity": "sha512-jQTChiCJteusULxjBp8+jftSQE5Obdl3k4cnmLA6WXtK6XFuWRnvVL7aCiBqaLPM8c4ph0S4tKna8XvmIwEnXQ==",
1282 | "requires": {
1283 | "forwarded": "0.1.2",
1284 | "ipaddr.js": "1.6.0"
1285 | }
1286 | },
1287 | "pseudomap": {
1288 | "version": "1.0.2",
1289 | "resolved": "https://registry.npmjs.org/pseudomap/-/pseudomap-1.0.2.tgz",
1290 | "integrity": "sha1-8FKijacOYYkX7wqKw0wa5aaChrM=",
1291 | "dev": true
1292 | },
1293 | "punycode": {
1294 | "version": "1.4.1",
1295 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz",
1296 | "integrity": "sha1-wNWmOycYgArY4esPpSachN1BhF4=",
1297 | "dev": true
1298 | },
1299 | "q": {
1300 | "version": "1.5.1",
1301 | "resolved": "https://registry.npmjs.org/q/-/q-1.5.1.tgz",
1302 | "integrity": "sha1-fjL3W0E4EpHQRhHxvxQQmsAGUdc=",
1303 | "dev": true
1304 | },
1305 | "qs": {
1306 | "version": "6.5.1",
1307 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.1.tgz",
1308 | "integrity": "sha512-eRzhrN1WSINYCDCbrz796z37LOe3m5tmW7RQf6oBntukAG1nmovJvhnwHHRMAfeoItc1m2Hk02WER2aQ/iqs+A=="
1309 | },
1310 | "querystring": {
1311 | "version": "0.2.0",
1312 | "resolved": "https://registry.npmjs.org/querystring/-/querystring-0.2.0.tgz",
1313 | "integrity": "sha1-sgmEkgO7Jd+CDadW50cAWHhSFiA="
1314 | },
1315 | "range-parser": {
1316 | "version": "1.2.0",
1317 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.0.tgz",
1318 | "integrity": "sha1-9JvmtIeJTdxA3MlKMi9hEJLgDV4="
1319 | },
1320 | "raw-body": {
1321 | "version": "2.3.2",
1322 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.3.2.tgz",
1323 | "integrity": "sha1-vNYMd9Prk83gBQKVw/N5OJvIj4k=",
1324 | "requires": {
1325 | "bytes": "3.0.0",
1326 | "http-errors": "1.6.2",
1327 | "iconv-lite": "0.4.19",
1328 | "unpipe": "1.0.0"
1329 | }
1330 | },
1331 | "readable-stream": {
1332 | "version": "2.3.5",
1333 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.5.tgz",
1334 | "integrity": "sha512-tK0yDhrkygt/knjowCUiWP9YdV7c5R+8cR0r/kt9ZhBU906Fs6RpQJCEilamRJj1Nx2rWI6LkW9gKqjTkshhEw==",
1335 | "dev": true,
1336 | "requires": {
1337 | "core-util-is": "1.0.2",
1338 | "inherits": "2.0.3",
1339 | "isarray": "1.0.0",
1340 | "process-nextick-args": "2.0.0",
1341 | "safe-buffer": "5.1.1",
1342 | "string_decoder": "1.0.3",
1343 | "util-deprecate": "1.0.2"
1344 | }
1345 | },
1346 | "remove-trailing-separator": {
1347 | "version": "1.1.0",
1348 | "resolved": "https://registry.npmjs.org/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz",
1349 | "integrity": "sha1-wkvOKig62tW8P1jg1IJJuSN52O8=",
1350 | "dev": true
1351 | },
1352 | "request": {
1353 | "version": "2.83.0",
1354 | "resolved": "https://registry.npmjs.org/request/-/request-2.83.0.tgz",
1355 | "integrity": "sha512-lR3gD69osqm6EYLk9wB/G1W/laGWjzH90t1vEa2xuxHD5KUrSzp9pUSfTm+YC5Nxt2T8nMPEvKlhbQayU7bgFw==",
1356 | "dev": true,
1357 | "requires": {
1358 | "aws-sign2": "0.7.0",
1359 | "aws4": "1.6.0",
1360 | "caseless": "0.12.0",
1361 | "combined-stream": "1.0.6",
1362 | "extend": "3.0.1",
1363 | "forever-agent": "0.6.1",
1364 | "form-data": "2.3.2",
1365 | "har-validator": "5.0.3",
1366 | "hawk": "6.0.2",
1367 | "http-signature": "1.2.0",
1368 | "is-typedarray": "1.0.0",
1369 | "isstream": "0.1.2",
1370 | "json-stringify-safe": "5.0.1",
1371 | "mime-types": "2.1.18",
1372 | "oauth-sign": "0.8.2",
1373 | "performance-now": "2.1.0",
1374 | "qs": "6.5.1",
1375 | "safe-buffer": "5.1.1",
1376 | "stringstream": "0.0.5",
1377 | "tough-cookie": "2.3.4",
1378 | "tunnel-agent": "0.6.0",
1379 | "uuid": "3.2.1"
1380 | }
1381 | },
1382 | "resolve-url": {
1383 | "version": "0.2.1",
1384 | "resolved": "https://registry.npmjs.org/resolve-url/-/resolve-url-0.2.1.tgz",
1385 | "integrity": "sha1-LGN/53yJOv0qZj/iGqkIAGjiBSo=",
1386 | "dev": true
1387 | },
1388 | "restore-cursor": {
1389 | "version": "1.0.1",
1390 | "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-1.0.1.tgz",
1391 | "integrity": "sha1-NGYfRohjJ/7SmRR5FSJS35LapUE=",
1392 | "dev": true,
1393 | "requires": {
1394 | "exit-hook": "1.1.1",
1395 | "onetime": "1.1.0"
1396 | }
1397 | },
1398 | "rimraf": {
1399 | "version": "2.4.5",
1400 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.4.5.tgz",
1401 | "integrity": "sha1-7nEM5dk6j9uFb7Xqj/Di11k0sto=",
1402 | "dev": true,
1403 | "optional": true,
1404 | "requires": {
1405 | "glob": "6.0.4"
1406 | }
1407 | },
1408 | "run-async": {
1409 | "version": "2.3.0",
1410 | "resolved": "https://registry.npmjs.org/run-async/-/run-async-2.3.0.tgz",
1411 | "integrity": "sha1-A3GrSuC91yDUFm19/aZP96RFpsA=",
1412 | "dev": true,
1413 | "requires": {
1414 | "is-promise": "2.1.0"
1415 | }
1416 | },
1417 | "rx": {
1418 | "version": "4.1.0",
1419 | "resolved": "https://registry.npmjs.org/rx/-/rx-4.1.0.tgz",
1420 | "integrity": "sha1-pfE/957zt0D+MKqAP7CfmIBdR4I=",
1421 | "dev": true
1422 | },
1423 | "safe-buffer": {
1424 | "version": "5.1.1",
1425 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.1.tgz",
1426 | "integrity": "sha512-kKvNJn6Mm93gAczWVJg7wH+wGYWNrDHdWvpUmHyEsgCtIwwo3bqPtV4tR5tuPaUhTOo/kvhVwd8XwwOllGYkbg=="
1427 | },
1428 | "safe-json-stringify": {
1429 | "version": "1.1.0",
1430 | "resolved": "https://registry.npmjs.org/safe-json-stringify/-/safe-json-stringify-1.1.0.tgz",
1431 | "integrity": "sha512-EzBtUaFH9bHYPc69wqjp0efJI/DPNHdFbGE3uIMn4sVbO0zx8vZ8cG4WKxQfOpUOKsQyGBiT2mTqnCw+6nLswA==",
1432 | "dev": true,
1433 | "optional": true
1434 | },
1435 | "send": {
1436 | "version": "0.16.1",
1437 | "resolved": "https://registry.npmjs.org/send/-/send-0.16.1.tgz",
1438 | "integrity": "sha512-ElCLJdJIKPk6ux/Hocwhk7NFHpI3pVm/IZOYWqUmoxcgeyM+MpxHHKhb8QmlJDX1pU6WrgaHBkVNm73Sv7uc2A==",
1439 | "requires": {
1440 | "debug": "2.6.9",
1441 | "depd": "1.1.2",
1442 | "destroy": "1.0.4",
1443 | "encodeurl": "1.0.2",
1444 | "escape-html": "1.0.3",
1445 | "etag": "1.8.1",
1446 | "fresh": "0.5.2",
1447 | "http-errors": "1.6.2",
1448 | "mime": "1.4.1",
1449 | "ms": "2.0.0",
1450 | "on-finished": "2.3.0",
1451 | "range-parser": "1.2.0",
1452 | "statuses": "1.3.1"
1453 | }
1454 | },
1455 | "serve-static": {
1456 | "version": "1.13.1",
1457 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.13.1.tgz",
1458 | "integrity": "sha512-hSMUZrsPa/I09VYFJwa627JJkNs0NrfL1Uzuup+GqHfToR2KcsXFymXSV90hoyw3M+msjFuQly+YzIH/q0MGlQ==",
1459 | "requires": {
1460 | "encodeurl": "1.0.2",
1461 | "escape-html": "1.0.3",
1462 | "parseurl": "1.3.2",
1463 | "send": "0.16.1"
1464 | }
1465 | },
1466 | "setprototypeof": {
1467 | "version": "1.1.0",
1468 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.0.tgz",
1469 | "integrity": "sha512-BvE/TwpZX4FXExxOxZyRGQQv651MSwmWKZGqvmPcRIjDqWub67kTKuIMx43cZZrS/cBBzwBcNDWoFxt2XEFIpQ=="
1470 | },
1471 | "shebang-command": {
1472 | "version": "1.2.0",
1473 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz",
1474 | "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=",
1475 | "dev": true,
1476 | "requires": {
1477 | "shebang-regex": "1.0.0"
1478 | }
1479 | },
1480 | "shebang-regex": {
1481 | "version": "1.0.0",
1482 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz",
1483 | "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=",
1484 | "dev": true
1485 | },
1486 | "sntp": {
1487 | "version": "2.1.0",
1488 | "resolved": "https://registry.npmjs.org/sntp/-/sntp-2.1.0.tgz",
1489 | "integrity": "sha512-FL1b58BDrqS3A11lJ0zEdnJ3UOKqVxawAkF3k7F0CVN7VQ34aZrV+G8BZ1WC9ZL7NyrwsW0oviwsWDgRuVYtJg==",
1490 | "dev": true,
1491 | "requires": {
1492 | "hoek": "4.2.1"
1493 | }
1494 | },
1495 | "spawn-sync": {
1496 | "version": "1.0.15",
1497 | "resolved": "https://registry.npmjs.org/spawn-sync/-/spawn-sync-1.0.15.tgz",
1498 | "integrity": "sha1-sAeZVX63+wyDdsKdROih6mfldHY=",
1499 | "dev": true,
1500 | "requires": {
1501 | "concat-stream": "1.6.1",
1502 | "os-shim": "0.1.3"
1503 | }
1504 | },
1505 | "sprintf-js": {
1506 | "version": "1.1.1",
1507 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.1.1.tgz",
1508 | "integrity": "sha1-Nr54Mgr+WAH2zqPueLblqrlA6gw=",
1509 | "dev": true
1510 | },
1511 | "sshpk": {
1512 | "version": "1.13.1",
1513 | "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.13.1.tgz",
1514 | "integrity": "sha1-US322mKHFEMW3EwY/hzx2UBzm+M=",
1515 | "dev": true,
1516 | "requires": {
1517 | "asn1": "0.2.3",
1518 | "assert-plus": "1.0.0",
1519 | "bcrypt-pbkdf": "1.0.1",
1520 | "dashdash": "1.14.1",
1521 | "ecc-jsbn": "0.1.1",
1522 | "getpass": "0.1.7",
1523 | "jsbn": "0.1.1",
1524 | "tweetnacl": "0.14.5"
1525 | }
1526 | },
1527 | "stack-trace": {
1528 | "version": "0.0.10",
1529 | "resolved": "https://registry.npmjs.org/stack-trace/-/stack-trace-0.0.10.tgz",
1530 | "integrity": "sha1-VHxws0fo0ytOEI6hoqFZ5f3eGcA=",
1531 | "dev": true
1532 | },
1533 | "statuses": {
1534 | "version": "1.3.1",
1535 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.3.1.tgz",
1536 | "integrity": "sha1-+vUbnrdKrvOzrPStX2Gr8ky3uT4="
1537 | },
1538 | "string-width": {
1539 | "version": "1.0.2",
1540 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz",
1541 | "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=",
1542 | "dev": true,
1543 | "requires": {
1544 | "code-point-at": "1.1.0",
1545 | "is-fullwidth-code-point": "1.0.0",
1546 | "strip-ansi": "3.0.1"
1547 | }
1548 | },
1549 | "string_decoder": {
1550 | "version": "1.0.3",
1551 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.0.3.tgz",
1552 | "integrity": "sha512-4AH6Z5fzNNBcH+6XDMfA/BTt87skxqJlO0lAh3Dker5zThcAxG6mKz+iGu308UKoPPQ8Dcqx/4JhujzltRa+hQ==",
1553 | "dev": true,
1554 | "requires": {
1555 | "safe-buffer": "5.1.1"
1556 | }
1557 | },
1558 | "stringstream": {
1559 | "version": "0.0.5",
1560 | "resolved": "https://registry.npmjs.org/stringstream/-/stringstream-0.0.5.tgz",
1561 | "integrity": "sha1-TkhM1N5aC7vuGORjB3EKioFiGHg=",
1562 | "dev": true
1563 | },
1564 | "strip-ansi": {
1565 | "version": "3.0.1",
1566 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz",
1567 | "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=",
1568 | "dev": true,
1569 | "requires": {
1570 | "ansi-regex": "2.1.1"
1571 | }
1572 | },
1573 | "superagent": {
1574 | "version": "3.8.2",
1575 | "resolved": "https://registry.npmjs.org/superagent/-/superagent-3.8.2.tgz",
1576 | "integrity": "sha512-gVH4QfYHcY3P0f/BZzavLreHW3T1v7hG9B+hpMQotGQqurOvhv87GcMCd6LWySmBuf+BDR44TQd0aISjVHLeNQ==",
1577 | "dev": true,
1578 | "requires": {
1579 | "component-emitter": "1.2.1",
1580 | "cookiejar": "2.1.1",
1581 | "debug": "3.1.0",
1582 | "extend": "3.0.1",
1583 | "form-data": "2.3.2",
1584 | "formidable": "1.1.1",
1585 | "methods": "1.1.2",
1586 | "mime": "1.4.1",
1587 | "qs": "6.5.1",
1588 | "readable-stream": "2.3.5"
1589 | },
1590 | "dependencies": {
1591 | "debug": {
1592 | "version": "3.1.0",
1593 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz",
1594 | "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==",
1595 | "dev": true,
1596 | "requires": {
1597 | "ms": "2.0.0"
1598 | }
1599 | }
1600 | }
1601 | },
1602 | "supports-color": {
1603 | "version": "2.0.0",
1604 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz",
1605 | "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=",
1606 | "dev": true
1607 | },
1608 | "tabtab": {
1609 | "version": "2.2.2",
1610 | "resolved": "https://registry.npmjs.org/tabtab/-/tabtab-2.2.2.tgz",
1611 | "integrity": "sha1-egR/FDsBC0y9MfhX6ClhUSy/ThQ=",
1612 | "dev": true,
1613 | "requires": {
1614 | "debug": "2.6.9",
1615 | "inquirer": "1.2.3",
1616 | "lodash.difference": "4.5.0",
1617 | "lodash.uniq": "4.5.0",
1618 | "minimist": "1.2.0",
1619 | "mkdirp": "0.5.1",
1620 | "npmlog": "2.0.4",
1621 | "object-assign": "4.1.1"
1622 | },
1623 | "dependencies": {
1624 | "minimist": {
1625 | "version": "1.2.0",
1626 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz",
1627 | "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ=",
1628 | "dev": true
1629 | }
1630 | }
1631 | },
1632 | "through": {
1633 | "version": "2.3.8",
1634 | "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz",
1635 | "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=",
1636 | "dev": true
1637 | },
1638 | "tmp": {
1639 | "version": "0.0.29",
1640 | "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.29.tgz",
1641 | "integrity": "sha1-8lEl/w3Z2jzLDC3Tce4SiLuRKMA=",
1642 | "dev": true,
1643 | "requires": {
1644 | "os-tmpdir": "1.0.2"
1645 | }
1646 | },
1647 | "tough-cookie": {
1648 | "version": "2.3.4",
1649 | "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.3.4.tgz",
1650 | "integrity": "sha512-TZ6TTfI5NtZnuyy/Kecv+CnoROnyXn2DN97LontgQpCwsX2XyLYCC0ENhYkehSOwAp8rTQKc/NUIF7BkQ5rKLA==",
1651 | "dev": true,
1652 | "requires": {
1653 | "punycode": "1.4.1"
1654 | }
1655 | },
1656 | "traverson": {
1657 | "version": "6.0.3",
1658 | "resolved": "https://registry.npmjs.org/traverson/-/traverson-6.0.3.tgz",
1659 | "integrity": "sha1-W8phq3NAHcpmRXmk6yuFH2R78to=",
1660 | "dev": true,
1661 | "requires": {
1662 | "jsonpath-plus": "0.16.0",
1663 | "minilog": "3.1.0",
1664 | "request": "2.83.0",
1665 | "resolve-url": "0.2.1",
1666 | "superagent": "3.8.2",
1667 | "underscore.string": "3.3.4",
1668 | "url-template": "2.0.8"
1669 | }
1670 | },
1671 | "traverson-hal": {
1672 | "version": "6.0.0",
1673 | "resolved": "https://registry.npmjs.org/traverson-hal/-/traverson-hal-6.0.0.tgz",
1674 | "integrity": "sha1-JTk1/WlWjeBlWESHXH+HQymRUmw=",
1675 | "dev": true,
1676 | "requires": {
1677 | "halfred": "1.0.0"
1678 | }
1679 | },
1680 | "traverson-promise": {
1681 | "version": "0.0.9",
1682 | "resolved": "https://registry.npmjs.org/traverson-promise/-/traverson-promise-0.0.9.tgz",
1683 | "integrity": "sha1-/lC3ailhFdAMW/Jr3eCDaPXTqX0=",
1684 | "dev": true,
1685 | "requires": {
1686 | "bluebird": "3.5.1",
1687 | "traverson": "6.0.3"
1688 | }
1689 | },
1690 | "tunnel-agent": {
1691 | "version": "0.6.0",
1692 | "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz",
1693 | "integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=",
1694 | "dev": true,
1695 | "requires": {
1696 | "safe-buffer": "5.1.1"
1697 | }
1698 | },
1699 | "tweetnacl": {
1700 | "version": "0.14.5",
1701 | "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz",
1702 | "integrity": "sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q=",
1703 | "dev": true,
1704 | "optional": true
1705 | },
1706 | "type-is": {
1707 | "version": "1.6.16",
1708 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.16.tgz",
1709 | "integrity": "sha512-HRkVv/5qY2G6I8iab9cI7v1bOIdhm94dVjQCPFElW9W+3GeDOSHmy2EBYe4VTApuzolPcmgFTN3ftVJRKR2J9Q==",
1710 | "requires": {
1711 | "media-typer": "0.3.0",
1712 | "mime-types": "2.1.18"
1713 | }
1714 | },
1715 | "typedarray": {
1716 | "version": "0.0.6",
1717 | "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz",
1718 | "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=",
1719 | "dev": true
1720 | },
1721 | "underscore": {
1722 | "version": "1.8.3",
1723 | "resolved": "https://registry.npmjs.org/underscore/-/underscore-1.8.3.tgz",
1724 | "integrity": "sha1-Tz+1OxBuYJf8+ctBCfKl6b36UCI=",
1725 | "dev": true
1726 | },
1727 | "underscore.string": {
1728 | "version": "3.3.4",
1729 | "resolved": "https://registry.npmjs.org/underscore.string/-/underscore.string-3.3.4.tgz",
1730 | "integrity": "sha1-LCo/n4PmR2L9xF5s6sZRQoZCE9s=",
1731 | "dev": true,
1732 | "requires": {
1733 | "sprintf-js": "1.1.1",
1734 | "util-deprecate": "1.0.2"
1735 | }
1736 | },
1737 | "unixify": {
1738 | "version": "1.0.0",
1739 | "resolved": "https://registry.npmjs.org/unixify/-/unixify-1.0.0.tgz",
1740 | "integrity": "sha1-OmQcjC/7zk2mg6XHDwOkYpQMIJA=",
1741 | "dev": true,
1742 | "requires": {
1743 | "normalize-path": "2.1.1"
1744 | }
1745 | },
1746 | "unpipe": {
1747 | "version": "1.0.0",
1748 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
1749 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw="
1750 | },
1751 | "url-join": {
1752 | "version": "4.0.0",
1753 | "resolved": "https://registry.npmjs.org/url-join/-/url-join-4.0.0.tgz",
1754 | "integrity": "sha1-TTNA6AfTdzvamZH4MFrNzCpmXSo=",
1755 | "dev": true
1756 | },
1757 | "url-template": {
1758 | "version": "2.0.8",
1759 | "resolved": "https://registry.npmjs.org/url-template/-/url-template-2.0.8.tgz",
1760 | "integrity": "sha1-/FZaPMy/93MMd19WQflVV5FDnyE=",
1761 | "dev": true
1762 | },
1763 | "util-deprecate": {
1764 | "version": "1.0.2",
1765 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
1766 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=",
1767 | "dev": true
1768 | },
1769 | "utils-merge": {
1770 | "version": "1.0.1",
1771 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
1772 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM="
1773 | },
1774 | "uuid": {
1775 | "version": "3.2.1",
1776 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.2.1.tgz",
1777 | "integrity": "sha512-jZnMwlb9Iku/O3smGWvZhauCf6cvvpKi4BKRiliS3cxnI+Gz9j5MEpTz2UFuXiKPJocb7gnsLHwiS05ige5BEA==",
1778 | "dev": true
1779 | },
1780 | "vary": {
1781 | "version": "1.1.2",
1782 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
1783 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw="
1784 | },
1785 | "verror": {
1786 | "version": "1.10.0",
1787 | "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz",
1788 | "integrity": "sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA=",
1789 | "dev": true,
1790 | "requires": {
1791 | "assert-plus": "1.0.0",
1792 | "core-util-is": "1.0.2",
1793 | "extsprintf": "1.3.0"
1794 | }
1795 | },
1796 | "which": {
1797 | "version": "1.3.0",
1798 | "resolved": "https://registry.npmjs.org/which/-/which-1.3.0.tgz",
1799 | "integrity": "sha512-xcJpopdamTuY5duC/KnTTNBraPK54YwpenP4lzxU8H91GudWpFv38u0CKjclE1Wi2EH2EDz5LRcHcKbCIzqGyg==",
1800 | "dev": true,
1801 | "requires": {
1802 | "isexe": "2.0.0"
1803 | }
1804 | },
1805 | "winston": {
1806 | "version": "2.4.0",
1807 | "resolved": "https://registry.npmjs.org/winston/-/winston-2.4.0.tgz",
1808 | "integrity": "sha1-gIBQuT1SZh7Z+2wms/DIJnCLCu4=",
1809 | "dev": true,
1810 | "requires": {
1811 | "async": "1.0.0",
1812 | "colors": "1.0.3",
1813 | "cycle": "1.0.3",
1814 | "eyes": "0.1.8",
1815 | "isstream": "0.1.2",
1816 | "stack-trace": "0.0.10"
1817 | },
1818 | "dependencies": {
1819 | "colors": {
1820 | "version": "1.0.3",
1821 | "resolved": "https://registry.npmjs.org/colors/-/colors-1.0.3.tgz",
1822 | "integrity": "sha1-BDP0TYCWgP3rYO0mDxsMJi6CpAs=",
1823 | "dev": true
1824 | }
1825 | }
1826 | },
1827 | "wrappy": {
1828 | "version": "1.0.2",
1829 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
1830 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=",
1831 | "dev": true
1832 | },
1833 | "yallist": {
1834 | "version": "2.1.2",
1835 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-2.1.2.tgz",
1836 | "integrity": "sha1-HBH5IY8HYImkfdUS+TxmmaaoHVI=",
1837 | "dev": true
1838 | }
1839 | }
1840 | }
1841 |
--------------------------------------------------------------------------------