├── .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 [![Build Status](https://travis-ci.org/mateuszsokola/contract-tests-example.svg?branch=master)](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`] = `"

Not Found

Proceed to job listing
"`; 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`] = `""`; 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 | --------------------------------------------------------------------------------