├── .eslintrc
├── .gitignore
├── .npmignore
├── .travis.yml
├── LICENSE
├── README.md
├── docs
└── demo.gif
├── package-lock.json
├── package.json
└── src
├── cli.js
├── commands
├── approve.js
└── test.js
├── index.js
└── utils
├── debug.js
├── image.js
├── options.js
└── page.js
/.eslintrc:
--------------------------------------------------------------------------------
1 | {
2 | "extends": ["standard"]
3 | }
4 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | /node_modules/
2 | /styleguide-visual/
3 |
--------------------------------------------------------------------------------
/.npmignore:
--------------------------------------------------------------------------------
1 | /docs/
2 | /styleguide-visual/
3 | /.eslintrc
4 |
--------------------------------------------------------------------------------
/.travis.yml:
--------------------------------------------------------------------------------
1 | language: node_js
2 | node_js:
3 | - '8'
4 |
5 | cache:
6 | directories:
7 | - node_modules
8 |
9 | deploy:
10 | provider: npm
11 | email: unindented@gmail.com
12 | api_key:
13 | secure: Z9Gw3V5YzYQSzEmau6MN/Co+jXTY2lfyfpsrkXLRYfdJXZ+6Bae/gkJipOwJbEluW/fAp1xlxZaQo/bmabtc1buednAQq5nu5ElQac7OO90tcWAk8kUM6I1l6S1ZlnJXknHCO9dGFAGhd611sLLPbo2Z46UhyhHrQIFuqf4AKy9xpGeXUfYZu5vjO7FHRAT4Mhqa1bR45Z49vRrtMEj4WB7KsKKkAYH54OTcpvRLEKjaUy/019UL5bX3FAjD0T8rUzyUkharTpXVRa6KHoJkIXOFuwHmQl2wQ5WwoDAgrZgqNZ0YpKVDCo1WdO8JREMsL83DkoESG+H+swnnNCBgDbCEVzYNpNnIB4DdmoGjRXNfTv8oGdd878WWMzhMXza8LqLg9gLgDhVRVXAeRdokX3+9KwBgLffgmVuWJUXujh/6yOCc0fGQ1bODE747Gu9Nc5O6krhtwS7lFZzc4LwAFyaI6Z0hIU3AdOm0SvZnX2Rw/SrzWNZkz5JSSoSHySJKgVtkIY65mXaknEnunZlN9MKQc+Ba9v8BsMsYc6LOku3rZ8f2X9FpIjMfUfvyqTVnY/iTqmSmf79xTB62Cf9Zg6X/yXaJ1IatFNDeKjeUM54XYBqeaEdqi7JImLDQ9aEDEKte4GQe4+bAe3QDjY6LxJWpk/Ti5HuND0u8QZ7nbQM=
14 | on:
15 | tags: true
16 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | Copyright (c) 2018 Daniel Perez Alvarez
2 |
3 | Permission is hereby granted, free of charge, to any person obtaining a copy of
4 | this software and associated documentation files (the "Software"), to deal in
5 | the Software without restriction, including without limitation the rights to
6 | use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
7 | of the Software, and to permit persons to whom the Software is furnished to do
8 | so, subject to the following conditions:
9 |
10 | The above copyright notice and this permission notice shall be included in all
11 | copies or substantial portions of the Software.
12 |
13 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
14 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
15 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
16 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
17 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
18 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
19 | SOFTWARE.
20 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Visual Testing for React Styleguidist [](https://www.npmjs.com/package/react-styleguidist-visual) [](http://travis-ci.org/unindented/react-styleguidist-visual)
2 |
3 | Allows you to do easy visual diffing of your [React Styleguidist](https://react-styleguidist.js.org/) examples.
4 |
5 | 
6 |
7 | ## Installation
8 |
9 | Add the dependency to your project:
10 |
11 | ```
12 | $ npm install --save-dev react-styleguidist-visual
13 | ```
14 |
15 | ## Usage
16 |
17 | Point the tool to your styleguide:
18 |
19 | ```
20 | $ npx styleguidist-visual test --url "https://react-styleguidist.js.org/examples/basic/"
21 | ```
22 |
23 | You can also test against your local style guide.
24 | The following command will first build the style guide and then run the visual test.
25 |
26 | ```
27 | $ npx styleguidist build && styleguidist-visual test --url \"file://$(pwd)/styleguide/index.html\"
28 | ```
29 |
30 | The first time you run the tool, it will create reference screenshots for all examples in your styleguide, and store them in the `styleguide-visual` folder. If you run the same command again, it will take new screenshots, compare them to the reference ones, and show you the differences between them.
31 |
32 | If the new screenshots look good, you can promote them to be the new reference files by running:
33 |
34 | ```
35 | $ npx styleguidist-visual approve
36 | ```
37 |
38 | ### Options
39 |
40 | You can see all possible options by appending the `--help` argument to any command:
41 |
42 | ```
43 | $ npx styleguidist-visual --help
44 | $ npx styleguidist-visual test --help
45 | $ npx styleguidist-visual approve --help
46 | ```
47 |
48 | ### Action States
49 |
50 | You can capture screenshots after simulating an action, by providing a `JSON.stringify`ed list of actions as props to the component wrapper like this:
51 |
52 | ````md
53 | ```js { "props": { "data-action-states": "[{\"action\":\"none\"},{\"action\":\"hover\",\"selector\":\".my-button\",\"wait\":\"1000\"},{\"action\":\"focus\",\"selector\":\".my-button\"},{\"action\":\"keyPress\",\"key\":\"Tab\"}]" } }
54 |
55 | ```
56 | ````
57 |
58 | Available actions are:
59 |
60 | - `none` - Capture the component without performing an action.
61 | - `hover` - Provide a `selector` to hover over.
62 | - `focus` - Provide a `selector` to focus on.
63 | - `click` - Provide a `selector` to click on.
64 | - `mouseDown` - Provide a `selector` to mouse down on.
65 | - `keyPress` - Provide a [`key`](https://github.com/GoogleChrome/puppeteer/blob/v1.4.0/lib/USKeyboardLayout.js) to press.
66 | - `wait` - An optional time in ms to wait between performing the action and snapping the screenshot.
67 |
68 | ### Debugging
69 |
70 | Use the `DEBUG` environment variable to see debugging statements:
71 |
72 | ```
73 | $ DEBUG=react-styleguidist-visual npx styleguidist-visual test --url "https://react-styleguidist.js.org/examples/basic/"
74 | ```
75 |
76 | ## Meta
77 |
78 | - Code: `git clone git://github.com/unindented/react-styleguidist-visual.git`
79 | - Home:
80 |
81 | ## Contributors
82 |
83 | - Daniel Perez Alvarez ([unindented@gmail.com](mailto:unindented@gmail.com))
84 |
85 | ## License
86 |
87 | Copyright (c) 2018 Daniel Perez Alvarez ([unindented.org](https://unindented.org/)). This is free software, and may be redistributed under the terms specified in the LICENSE file.
88 |
--------------------------------------------------------------------------------
/docs/demo.gif:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/unindented/react-styleguidist-visual/8e8f5cc058ce8e55a945031b2de23303735f9b48/docs/demo.gif
--------------------------------------------------------------------------------
/package-lock.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "react-styleguidist-visual",
3 | "version": "0.9.0",
4 | "lockfileVersion": 1,
5 | "requires": true,
6 | "dependencies": {
7 | "acorn": {
8 | "version": "5.6.2",
9 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-5.6.2.tgz",
10 | "integrity": "sha512-zUzo1E5dI2Ey8+82egfnttyMlMZ2y0D8xOCO3PNPPlYXpl8NZvF6Qk9L9BEtJs+43FqEmfBViDqc5d1ckRDguw==",
11 | "dev": true
12 | },
13 | "acorn-jsx": {
14 | "version": "3.0.1",
15 | "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-3.0.1.tgz",
16 | "integrity": "sha1-r9+UiPsezvyDSPb7IvRk4ypYs2s=",
17 | "dev": true,
18 | "requires": {
19 | "acorn": "^3.0.4"
20 | },
21 | "dependencies": {
22 | "acorn": {
23 | "version": "3.3.0",
24 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-3.3.0.tgz",
25 | "integrity": "sha1-ReN/s56No/JbruP/U2niu18iAXo=",
26 | "dev": true
27 | }
28 | }
29 | },
30 | "agent-base": {
31 | "version": "4.2.0",
32 | "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-4.2.0.tgz",
33 | "integrity": "sha512-c+R/U5X+2zz2+UCrCFv6odQzJdoqI+YecuhnAJLa1zYaMc13zPfwMwZrr91Pd1DYNo/yPRbiM4WVf9whgwFsIg==",
34 | "requires": {
35 | "es6-promisify": "^5.0.0"
36 | }
37 | },
38 | "ajv": {
39 | "version": "5.5.2",
40 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-5.5.2.tgz",
41 | "integrity": "sha1-c7Xuyj+rZT49P5Qis0GtQiBdyWU=",
42 | "dev": true,
43 | "requires": {
44 | "co": "^4.6.0",
45 | "fast-deep-equal": "^1.0.0",
46 | "fast-json-stable-stringify": "^2.0.0",
47 | "json-schema-traverse": "^0.3.0"
48 | }
49 | },
50 | "ajv-keywords": {
51 | "version": "2.1.1",
52 | "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-2.1.1.tgz",
53 | "integrity": "sha1-YXmX/F9gV2iUxDX5QNgZ4TW4B2I=",
54 | "dev": true
55 | },
56 | "ansi-escapes": {
57 | "version": "2.0.0",
58 | "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-2.0.0.tgz",
59 | "integrity": "sha1-W65SvkJIeN2Xg+iRDj/Cki6DyBs="
60 | },
61 | "ansi-regex": {
62 | "version": "3.0.0",
63 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz",
64 | "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg="
65 | },
66 | "ansi-styles": {
67 | "version": "3.2.1",
68 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
69 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
70 | "requires": {
71 | "color-convert": "^1.9.0"
72 | }
73 | },
74 | "app-path": {
75 | "version": "2.2.0",
76 | "resolved": "https://registry.npmjs.org/app-path/-/app-path-2.2.0.tgz",
77 | "integrity": "sha1-KvXCtUSkDhX8GsVVSDFDl0YIRdA=",
78 | "requires": {
79 | "execa": "^0.4.0"
80 | }
81 | },
82 | "argparse": {
83 | "version": "1.0.10",
84 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz",
85 | "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==",
86 | "dev": true,
87 | "requires": {
88 | "sprintf-js": "~1.0.2"
89 | }
90 | },
91 | "array-union": {
92 | "version": "1.0.2",
93 | "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz",
94 | "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=",
95 | "dev": true,
96 | "requires": {
97 | "array-uniq": "^1.0.1"
98 | }
99 | },
100 | "array-uniq": {
101 | "version": "1.0.3",
102 | "resolved": "https://registry.npmjs.org/array-uniq/-/array-uniq-1.0.3.tgz",
103 | "integrity": "sha1-r2rId6Jcx/dOBYiUdThY39sk/bY=",
104 | "dev": true
105 | },
106 | "arrify": {
107 | "version": "1.0.1",
108 | "resolved": "https://registry.npmjs.org/arrify/-/arrify-1.0.1.tgz",
109 | "integrity": "sha1-iYUI2iIm84DfkEcoRWhJwVAaSw0=",
110 | "dev": true
111 | },
112 | "async-limiter": {
113 | "version": "1.0.0",
114 | "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.0.tgz",
115 | "integrity": "sha512-jp/uFnooOiO+L211eZOoSyzpOITMXx1rBITauYykG3BRYPu8h0UcxsPNB04RR5vo4Tyz3+ay17tR6JVf9qzYWg=="
116 | },
117 | "babel-code-frame": {
118 | "version": "6.26.0",
119 | "resolved": "https://registry.npmjs.org/babel-code-frame/-/babel-code-frame-6.26.0.tgz",
120 | "integrity": "sha1-Y/1D99weO7fONZR9uP42mj9Yx0s=",
121 | "dev": true,
122 | "requires": {
123 | "chalk": "^1.1.3",
124 | "esutils": "^2.0.2",
125 | "js-tokens": "^3.0.2"
126 | },
127 | "dependencies": {
128 | "ansi-regex": {
129 | "version": "2.1.1",
130 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz",
131 | "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=",
132 | "dev": true
133 | },
134 | "ansi-styles": {
135 | "version": "2.2.1",
136 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz",
137 | "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=",
138 | "dev": true
139 | },
140 | "chalk": {
141 | "version": "1.1.3",
142 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz",
143 | "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=",
144 | "dev": true,
145 | "requires": {
146 | "ansi-styles": "^2.2.1",
147 | "escape-string-regexp": "^1.0.2",
148 | "has-ansi": "^2.0.0",
149 | "strip-ansi": "^3.0.0",
150 | "supports-color": "^2.0.0"
151 | }
152 | },
153 | "strip-ansi": {
154 | "version": "3.0.1",
155 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz",
156 | "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=",
157 | "dev": true,
158 | "requires": {
159 | "ansi-regex": "^2.0.0"
160 | }
161 | },
162 | "supports-color": {
163 | "version": "2.0.0",
164 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz",
165 | "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=",
166 | "dev": true
167 | }
168 | }
169 | },
170 | "balanced-match": {
171 | "version": "1.0.0",
172 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz",
173 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c="
174 | },
175 | "base64-js": {
176 | "version": "1.2.0",
177 | "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.2.0.tgz",
178 | "integrity": "sha1-o5mS1yNYSBGYK+XikLtqU9hnAPE="
179 | },
180 | "brace-expansion": {
181 | "version": "1.1.11",
182 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
183 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
184 | "requires": {
185 | "balanced-match": "^1.0.0",
186 | "concat-map": "0.0.1"
187 | }
188 | },
189 | "buffer-from": {
190 | "version": "1.1.0",
191 | "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.0.tgz",
192 | "integrity": "sha512-c5mRlguI/Pe2dSZmpER62rSCu0ryKmWddzRYsuXc50U2/g8jMOulc31VZMa4mYx31U5xsmSOpDCgH88Vl9cDGQ=="
193 | },
194 | "builtin-modules": {
195 | "version": "1.1.1",
196 | "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-1.1.1.tgz",
197 | "integrity": "sha1-Jw8HbFpywC9bZaR9+Uxf46J4iS8=",
198 | "dev": true
199 | },
200 | "caller-path": {
201 | "version": "0.1.0",
202 | "resolved": "https://registry.npmjs.org/caller-path/-/caller-path-0.1.0.tgz",
203 | "integrity": "sha1-lAhe9jWB7NPaqSREqP6U6CV3dR8=",
204 | "dev": true,
205 | "requires": {
206 | "callsites": "^0.2.0"
207 | }
208 | },
209 | "callsites": {
210 | "version": "0.2.0",
211 | "resolved": "https://registry.npmjs.org/callsites/-/callsites-0.2.0.tgz",
212 | "integrity": "sha1-r6uWJikQp/M8GaV3WCXGnzTjUMo=",
213 | "dev": true
214 | },
215 | "camelcase": {
216 | "version": "4.1.0",
217 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-4.1.0.tgz",
218 | "integrity": "sha1-1UVjW+HjPFQmScaRc+Xeas+uNN0="
219 | },
220 | "chalk": {
221 | "version": "2.4.1",
222 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.1.tgz",
223 | "integrity": "sha512-ObN6h1v2fTJSmUXoS3nMQ92LbDK9be4TV+6G+omQlGJFdcUX5heKi1LZ1YnRMIgwTLEj3E24bT6tYni50rlCfQ==",
224 | "requires": {
225 | "ansi-styles": "^3.2.1",
226 | "escape-string-regexp": "^1.0.5",
227 | "supports-color": "^5.3.0"
228 | }
229 | },
230 | "chardet": {
231 | "version": "0.4.2",
232 | "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.4.2.tgz",
233 | "integrity": "sha1-tUc7M9yXxCTl2Y3IfVXU2KKci/I=",
234 | "dev": true
235 | },
236 | "circular-json": {
237 | "version": "0.3.3",
238 | "resolved": "https://registry.npmjs.org/circular-json/-/circular-json-0.3.3.tgz",
239 | "integrity": "sha512-UZK3NBx2Mca+b5LsG7bY183pHWt5Y1xts4P3Pz7ENTwGVnJOUWbRb3ocjvX7hx9tq/yTAdclXm9sZ38gNuem4A==",
240 | "dev": true
241 | },
242 | "cli-cursor": {
243 | "version": "2.1.0",
244 | "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz",
245 | "integrity": "sha1-s12sN2R5+sw+lHR9QdDQ9SOP/LU=",
246 | "requires": {
247 | "restore-cursor": "^2.0.0"
248 | }
249 | },
250 | "cli-spinners": {
251 | "version": "1.3.1",
252 | "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-1.3.1.tgz",
253 | "integrity": "sha512-1QL4544moEsDVH9T/l6Cemov/37iv1RtoKf7NJ04A60+4MREXNfx/QvavbH6QoGdsD4N4Mwy49cmaINR/o2mdg=="
254 | },
255 | "cli-width": {
256 | "version": "2.2.0",
257 | "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.2.0.tgz",
258 | "integrity": "sha1-/xnt6Kml5XkyQUewwR8PvLq+1jk=",
259 | "dev": true
260 | },
261 | "cliui": {
262 | "version": "4.1.0",
263 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-4.1.0.tgz",
264 | "integrity": "sha512-4FG+RSG9DL7uEwRUZXZn3SS34DiDPfzP0VOiEwtUWlE+AR2EIg+hSyvrIgUUfhdgR/UkAeW2QHgeP+hWrXs7jQ==",
265 | "requires": {
266 | "string-width": "^2.1.1",
267 | "strip-ansi": "^4.0.0",
268 | "wrap-ansi": "^2.0.0"
269 | }
270 | },
271 | "clone": {
272 | "version": "1.0.4",
273 | "resolved": "https://registry.npmjs.org/clone/-/clone-1.0.4.tgz",
274 | "integrity": "sha1-2jCcwmPfFZlMaIypAheco8fNfH4="
275 | },
276 | "co": {
277 | "version": "4.6.0",
278 | "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz",
279 | "integrity": "sha1-bqa989hTrlTMuOR7+gvz+QMfsYQ=",
280 | "dev": true
281 | },
282 | "code-point-at": {
283 | "version": "1.1.0",
284 | "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz",
285 | "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c="
286 | },
287 | "color-convert": {
288 | "version": "1.9.1",
289 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.1.tgz",
290 | "integrity": "sha512-mjGanIiwQJskCC18rPR6OmrZ6fm2Lc7PeGFYwCmy5J34wC6F1PzdGL6xeMfmgicfYcNLGuVFA3WzXtIDCQSZxQ==",
291 | "requires": {
292 | "color-name": "^1.1.1"
293 | }
294 | },
295 | "color-name": {
296 | "version": "1.1.3",
297 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
298 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU="
299 | },
300 | "concat-map": {
301 | "version": "0.0.1",
302 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
303 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s="
304 | },
305 | "concat-stream": {
306 | "version": "1.6.2",
307 | "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz",
308 | "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==",
309 | "requires": {
310 | "buffer-from": "^1.0.0",
311 | "inherits": "^2.0.3",
312 | "readable-stream": "^2.2.2",
313 | "typedarray": "^0.0.6"
314 | }
315 | },
316 | "contains-path": {
317 | "version": "0.1.0",
318 | "resolved": "https://registry.npmjs.org/contains-path/-/contains-path-0.1.0.tgz",
319 | "integrity": "sha1-/ozxhP9mcLa67wGp1IYaXL7EEgo=",
320 | "dev": true
321 | },
322 | "core-util-is": {
323 | "version": "1.0.2",
324 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz",
325 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac="
326 | },
327 | "cross-spawn": {
328 | "version": "5.1.0",
329 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-5.1.0.tgz",
330 | "integrity": "sha1-6L0O/uWPz/b4+UUQoKVUu/ojVEk=",
331 | "requires": {
332 | "lru-cache": "^4.0.1",
333 | "shebang-command": "^1.2.0",
334 | "which": "^1.2.9"
335 | }
336 | },
337 | "cross-spawn-async": {
338 | "version": "2.2.5",
339 | "resolved": "https://registry.npmjs.org/cross-spawn-async/-/cross-spawn-async-2.2.5.tgz",
340 | "integrity": "sha1-hF/wwINKPe2dFg2sptOQkGuyiMw=",
341 | "requires": {
342 | "lru-cache": "^4.0.0",
343 | "which": "^1.2.8"
344 | }
345 | },
346 | "debug": {
347 | "version": "3.1.0",
348 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz",
349 | "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==",
350 | "requires": {
351 | "ms": "2.0.0"
352 | }
353 | },
354 | "decamelize": {
355 | "version": "1.2.0",
356 | "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz",
357 | "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA="
358 | },
359 | "deep-is": {
360 | "version": "0.1.3",
361 | "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.3.tgz",
362 | "integrity": "sha1-s2nW+128E+7PUk+RsHD+7cNXzzQ=",
363 | "dev": true
364 | },
365 | "defaults": {
366 | "version": "1.0.3",
367 | "resolved": "https://registry.npmjs.org/defaults/-/defaults-1.0.3.tgz",
368 | "integrity": "sha1-xlYFHpgX2f8I7YgUd/P+QBnz730=",
369 | "requires": {
370 | "clone": "^1.0.2"
371 | }
372 | },
373 | "del": {
374 | "version": "2.2.2",
375 | "resolved": "https://registry.npmjs.org/del/-/del-2.2.2.tgz",
376 | "integrity": "sha1-wSyYHQZ4RshLyvhiz/kw2Qf/0ag=",
377 | "dev": true,
378 | "requires": {
379 | "globby": "^5.0.0",
380 | "is-path-cwd": "^1.0.0",
381 | "is-path-in-cwd": "^1.0.0",
382 | "object-assign": "^4.0.1",
383 | "pify": "^2.0.0",
384 | "pinkie-promise": "^2.0.0",
385 | "rimraf": "^2.2.8"
386 | }
387 | },
388 | "doctrine": {
389 | "version": "2.1.0",
390 | "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz",
391 | "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==",
392 | "dev": true,
393 | "requires": {
394 | "esutils": "^2.0.2"
395 | }
396 | },
397 | "error-ex": {
398 | "version": "1.3.1",
399 | "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.1.tgz",
400 | "integrity": "sha1-+FWobOYa3E6GIcPNoh56dhLDqNw=",
401 | "dev": true,
402 | "requires": {
403 | "is-arrayish": "^0.2.1"
404 | }
405 | },
406 | "es6-promise": {
407 | "version": "4.2.4",
408 | "resolved": "https://registry.npmjs.org/es6-promise/-/es6-promise-4.2.4.tgz",
409 | "integrity": "sha512-/NdNZVJg+uZgtm9eS3O6lrOLYmQag2DjdEXuPaHlZ6RuVqgqaVZfgYCepEIKsLqwdQArOPtC3XzRLqGGfT8KQQ=="
410 | },
411 | "es6-promisify": {
412 | "version": "5.0.0",
413 | "resolved": "https://registry.npmjs.org/es6-promisify/-/es6-promisify-5.0.0.tgz",
414 | "integrity": "sha1-UQnWLz5W6pZ8S2NQWu8IKRyKUgM=",
415 | "requires": {
416 | "es6-promise": "^4.0.3"
417 | }
418 | },
419 | "escape-string-regexp": {
420 | "version": "1.0.5",
421 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz",
422 | "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ="
423 | },
424 | "eslint": {
425 | "version": "4.19.1",
426 | "resolved": "https://registry.npmjs.org/eslint/-/eslint-4.19.1.tgz",
427 | "integrity": "sha512-bT3/1x1EbZB7phzYu7vCr1v3ONuzDtX8WjuM9c0iYxe+cq+pwcKEoQjl7zd3RpC6YOLgnSy3cTN58M2jcoPDIQ==",
428 | "dev": true,
429 | "requires": {
430 | "ajv": "^5.3.0",
431 | "babel-code-frame": "^6.22.0",
432 | "chalk": "^2.1.0",
433 | "concat-stream": "^1.6.0",
434 | "cross-spawn": "^5.1.0",
435 | "debug": "^3.1.0",
436 | "doctrine": "^2.1.0",
437 | "eslint-scope": "^3.7.1",
438 | "eslint-visitor-keys": "^1.0.0",
439 | "espree": "^3.5.4",
440 | "esquery": "^1.0.0",
441 | "esutils": "^2.0.2",
442 | "file-entry-cache": "^2.0.0",
443 | "functional-red-black-tree": "^1.0.1",
444 | "glob": "^7.1.2",
445 | "globals": "^11.0.1",
446 | "ignore": "^3.3.3",
447 | "imurmurhash": "^0.1.4",
448 | "inquirer": "^3.0.6",
449 | "is-resolvable": "^1.0.0",
450 | "js-yaml": "^3.9.1",
451 | "json-stable-stringify-without-jsonify": "^1.0.1",
452 | "levn": "^0.3.0",
453 | "lodash": "^4.17.4",
454 | "minimatch": "^3.0.2",
455 | "mkdirp": "^0.5.1",
456 | "natural-compare": "^1.4.0",
457 | "optionator": "^0.8.2",
458 | "path-is-inside": "^1.0.2",
459 | "pluralize": "^7.0.0",
460 | "progress": "^2.0.0",
461 | "regexpp": "^1.0.1",
462 | "require-uncached": "^1.0.3",
463 | "semver": "^5.3.0",
464 | "strip-ansi": "^4.0.0",
465 | "strip-json-comments": "~2.0.1",
466 | "table": "4.0.2",
467 | "text-table": "~0.2.0"
468 | }
469 | },
470 | "eslint-config-standard": {
471 | "version": "11.0.0",
472 | "resolved": "https://registry.npmjs.org/eslint-config-standard/-/eslint-config-standard-11.0.0.tgz",
473 | "integrity": "sha512-oDdENzpViEe5fwuRCWla7AXQd++/oyIp8zP+iP9jiUPG6NBj3SHgdgtl/kTn00AjeN+1HNvavTKmYbMo+xMOlw==",
474 | "dev": true
475 | },
476 | "eslint-import-resolver-node": {
477 | "version": "0.3.2",
478 | "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.2.tgz",
479 | "integrity": "sha512-sfmTqJfPSizWu4aymbPr4Iidp5yKm8yDkHp+Ir3YiTHiiDfxh69mOUsmiqW6RZ9zRXFaF64GtYmN7e+8GHBv6Q==",
480 | "dev": true,
481 | "requires": {
482 | "debug": "^2.6.9",
483 | "resolve": "^1.5.0"
484 | },
485 | "dependencies": {
486 | "debug": {
487 | "version": "2.6.9",
488 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
489 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
490 | "dev": true,
491 | "requires": {
492 | "ms": "2.0.0"
493 | }
494 | }
495 | }
496 | },
497 | "eslint-module-utils": {
498 | "version": "2.2.0",
499 | "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.2.0.tgz",
500 | "integrity": "sha1-snA2LNiLGkitMIl2zn+lTphBF0Y=",
501 | "dev": true,
502 | "requires": {
503 | "debug": "^2.6.8",
504 | "pkg-dir": "^1.0.0"
505 | },
506 | "dependencies": {
507 | "debug": {
508 | "version": "2.6.9",
509 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
510 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
511 | "dev": true,
512 | "requires": {
513 | "ms": "2.0.0"
514 | }
515 | }
516 | }
517 | },
518 | "eslint-plugin-import": {
519 | "version": "2.12.0",
520 | "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.12.0.tgz",
521 | "integrity": "sha1-2tMXgSktZmSyUxf9BJ0uKy8CIF0=",
522 | "dev": true,
523 | "requires": {
524 | "contains-path": "^0.1.0",
525 | "debug": "^2.6.8",
526 | "doctrine": "1.5.0",
527 | "eslint-import-resolver-node": "^0.3.1",
528 | "eslint-module-utils": "^2.2.0",
529 | "has": "^1.0.1",
530 | "lodash": "^4.17.4",
531 | "minimatch": "^3.0.3",
532 | "read-pkg-up": "^2.0.0",
533 | "resolve": "^1.6.0"
534 | },
535 | "dependencies": {
536 | "debug": {
537 | "version": "2.6.9",
538 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
539 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
540 | "dev": true,
541 | "requires": {
542 | "ms": "2.0.0"
543 | }
544 | },
545 | "doctrine": {
546 | "version": "1.5.0",
547 | "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-1.5.0.tgz",
548 | "integrity": "sha1-N53Ocw9hZvds76TmcHoVmwLFpvo=",
549 | "dev": true,
550 | "requires": {
551 | "esutils": "^2.0.2",
552 | "isarray": "^1.0.0"
553 | }
554 | }
555 | }
556 | },
557 | "eslint-plugin-node": {
558 | "version": "6.0.1",
559 | "resolved": "https://registry.npmjs.org/eslint-plugin-node/-/eslint-plugin-node-6.0.1.tgz",
560 | "integrity": "sha512-Q/Cc2sW1OAISDS+Ji6lZS2KV4b7ueA/WydVWd1BECTQwVvfQy5JAi3glhINoKzoMnfnuRgNP+ZWKrGAbp3QDxw==",
561 | "dev": true,
562 | "requires": {
563 | "ignore": "^3.3.6",
564 | "minimatch": "^3.0.4",
565 | "resolve": "^1.3.3",
566 | "semver": "^5.4.1"
567 | }
568 | },
569 | "eslint-plugin-promise": {
570 | "version": "3.8.0",
571 | "resolved": "https://registry.npmjs.org/eslint-plugin-promise/-/eslint-plugin-promise-3.8.0.tgz",
572 | "integrity": "sha512-JiFL9UFR15NKpHyGii1ZcvmtIqa3UTwiDAGb8atSffe43qJ3+1czVGN6UtkklpcJ2DVnqvTMzEKRaJdBkAL2aQ==",
573 | "dev": true
574 | },
575 | "eslint-plugin-standard": {
576 | "version": "3.1.0",
577 | "resolved": "https://registry.npmjs.org/eslint-plugin-standard/-/eslint-plugin-standard-3.1.0.tgz",
578 | "integrity": "sha512-fVcdyuKRr0EZ4fjWl3c+gp1BANFJD1+RaWa2UPYfMZ6jCtp5RG00kSaXnK/dE5sYzt4kaWJ9qdxqUfc0d9kX0w==",
579 | "dev": true
580 | },
581 | "eslint-scope": {
582 | "version": "3.7.1",
583 | "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-3.7.1.tgz",
584 | "integrity": "sha1-PWPD7f2gLgbgGkUq2IyqzHzctug=",
585 | "dev": true,
586 | "requires": {
587 | "esrecurse": "^4.1.0",
588 | "estraverse": "^4.1.1"
589 | }
590 | },
591 | "eslint-visitor-keys": {
592 | "version": "1.0.0",
593 | "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.0.0.tgz",
594 | "integrity": "sha512-qzm/XxIbxm/FHyH341ZrbnMUpe+5Bocte9xkmFMzPMjRaZMcXww+MpBptFvtU+79L362nqiLhekCxCxDPaUMBQ==",
595 | "dev": true
596 | },
597 | "espree": {
598 | "version": "3.5.4",
599 | "resolved": "https://registry.npmjs.org/espree/-/espree-3.5.4.tgz",
600 | "integrity": "sha512-yAcIQxtmMiB/jL32dzEp2enBeidsB7xWPLNiw3IIkpVds1P+h7qF9YwJq1yUNzp2OKXgAprs4F61ih66UsoD1A==",
601 | "dev": true,
602 | "requires": {
603 | "acorn": "^5.5.0",
604 | "acorn-jsx": "^3.0.0"
605 | }
606 | },
607 | "esprima": {
608 | "version": "4.0.0",
609 | "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.0.tgz",
610 | "integrity": "sha512-oftTcaMu/EGrEIu904mWteKIv8vMuOgGYo7EhVJJN00R/EED9DCua/xxHRdYnKtcECzVg7xOWhflvJMnqcFZjw==",
611 | "dev": true
612 | },
613 | "esquery": {
614 | "version": "1.0.1",
615 | "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.0.1.tgz",
616 | "integrity": "sha512-SmiyZ5zIWH9VM+SRUReLS5Q8a7GxtRdxEBVZpm98rJM7Sb+A9DVCndXfkeFUd3byderg+EbDkfnevfCwynWaNA==",
617 | "dev": true,
618 | "requires": {
619 | "estraverse": "^4.0.0"
620 | }
621 | },
622 | "esrecurse": {
623 | "version": "4.2.1",
624 | "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.2.1.tgz",
625 | "integrity": "sha512-64RBB++fIOAXPw3P9cy89qfMlvZEXZkqqJkjqqXIvzP5ezRZjW+lPWjw35UX/3EhUPFYbg5ER4JYgDw4007/DQ==",
626 | "dev": true,
627 | "requires": {
628 | "estraverse": "^4.1.0"
629 | }
630 | },
631 | "estraverse": {
632 | "version": "4.2.0",
633 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.2.0.tgz",
634 | "integrity": "sha1-De4/7TH81GlhjOc0IJn8GvoL2xM=",
635 | "dev": true
636 | },
637 | "esutils": {
638 | "version": "2.0.2",
639 | "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.2.tgz",
640 | "integrity": "sha1-Cr9PHKpbyx96nYrMbepPqqBLrJs=",
641 | "dev": true
642 | },
643 | "execa": {
644 | "version": "0.4.0",
645 | "resolved": "https://registry.npmjs.org/execa/-/execa-0.4.0.tgz",
646 | "integrity": "sha1-TrZGejaglfq7KXD/nV4/t7zm68M=",
647 | "requires": {
648 | "cross-spawn-async": "^2.1.1",
649 | "is-stream": "^1.1.0",
650 | "npm-run-path": "^1.0.0",
651 | "object-assign": "^4.0.1",
652 | "path-key": "^1.0.0",
653 | "strip-eof": "^1.0.0"
654 | }
655 | },
656 | "external-editor": {
657 | "version": "2.2.0",
658 | "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-2.2.0.tgz",
659 | "integrity": "sha512-bSn6gvGxKt+b7+6TKEv1ZycHleA7aHhRHyAqJyp5pbUFuYYNIzpZnQDk7AsYckyWdEnTeAnay0aCy2aV6iTk9A==",
660 | "dev": true,
661 | "requires": {
662 | "chardet": "^0.4.0",
663 | "iconv-lite": "^0.4.17",
664 | "tmp": "^0.0.33"
665 | }
666 | },
667 | "extract-zip": {
668 | "version": "1.6.7",
669 | "resolved": "https://registry.npmjs.org/extract-zip/-/extract-zip-1.6.7.tgz",
670 | "integrity": "sha1-qEC0uK9kAyZMjbV/Txp0Mz74H+k=",
671 | "requires": {
672 | "concat-stream": "1.6.2",
673 | "debug": "2.6.9",
674 | "mkdirp": "0.5.1",
675 | "yauzl": "2.4.1"
676 | },
677 | "dependencies": {
678 | "debug": {
679 | "version": "2.6.9",
680 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
681 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
682 | "requires": {
683 | "ms": "2.0.0"
684 | }
685 | }
686 | }
687 | },
688 | "fast-deep-equal": {
689 | "version": "1.1.0",
690 | "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-1.1.0.tgz",
691 | "integrity": "sha1-wFNHeBfIa1HaqFPIHgWbcz0CNhQ=",
692 | "dev": true
693 | },
694 | "fast-json-stable-stringify": {
695 | "version": "2.0.0",
696 | "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.0.0.tgz",
697 | "integrity": "sha1-1RQsDK7msRifh9OnYREGT4bIu/I=",
698 | "dev": true
699 | },
700 | "fast-levenshtein": {
701 | "version": "2.0.6",
702 | "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz",
703 | "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=",
704 | "dev": true
705 | },
706 | "fd-slicer": {
707 | "version": "1.0.1",
708 | "resolved": "https://registry.npmjs.org/fd-slicer/-/fd-slicer-1.0.1.tgz",
709 | "integrity": "sha1-i1vL2ewyfFBBv5qwI/1nUPEXfmU=",
710 | "requires": {
711 | "pend": "~1.2.0"
712 | }
713 | },
714 | "figures": {
715 | "version": "2.0.0",
716 | "resolved": "https://registry.npmjs.org/figures/-/figures-2.0.0.tgz",
717 | "integrity": "sha1-OrGi0qYsi/tDGgyUy3l6L84nyWI=",
718 | "dev": true,
719 | "requires": {
720 | "escape-string-regexp": "^1.0.5"
721 | }
722 | },
723 | "file-entry-cache": {
724 | "version": "2.0.0",
725 | "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-2.0.0.tgz",
726 | "integrity": "sha1-w5KZDD5oR4PYOLjISkXYoEhFg2E=",
727 | "dev": true,
728 | "requires": {
729 | "flat-cache": "^1.2.1",
730 | "object-assign": "^4.0.1"
731 | }
732 | },
733 | "find-up": {
734 | "version": "2.1.0",
735 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz",
736 | "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c=",
737 | "requires": {
738 | "locate-path": "^2.0.0"
739 | }
740 | },
741 | "flat-cache": {
742 | "version": "1.3.0",
743 | "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-1.3.0.tgz",
744 | "integrity": "sha1-0wMLMrOBVPTjt+nHCfSQ9++XxIE=",
745 | "dev": true,
746 | "requires": {
747 | "circular-json": "^0.3.1",
748 | "del": "^2.0.2",
749 | "graceful-fs": "^4.1.2",
750 | "write": "^0.2.1"
751 | }
752 | },
753 | "fs-extra": {
754 | "version": "6.0.1",
755 | "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-6.0.1.tgz",
756 | "integrity": "sha512-GnyIkKhhzXZUWFCaJzvyDLEEgDkPfb4/TPvJCJVuS8MWZgoSsErf++QpiAlDnKFcqhRlm+tIOcencCjyJE6ZCA==",
757 | "requires": {
758 | "graceful-fs": "^4.1.2",
759 | "jsonfile": "^4.0.0",
760 | "universalify": "^0.1.0"
761 | }
762 | },
763 | "fs.realpath": {
764 | "version": "1.0.0",
765 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz",
766 | "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8="
767 | },
768 | "function-bind": {
769 | "version": "1.1.1",
770 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz",
771 | "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==",
772 | "dev": true
773 | },
774 | "functional-red-black-tree": {
775 | "version": "1.0.1",
776 | "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz",
777 | "integrity": "sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc=",
778 | "dev": true
779 | },
780 | "get-caller-file": {
781 | "version": "1.0.2",
782 | "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-1.0.2.tgz",
783 | "integrity": "sha1-9wLmMSfn4jHBYKgMFVSstw1QR+U="
784 | },
785 | "get-stream": {
786 | "version": "3.0.0",
787 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-3.0.0.tgz",
788 | "integrity": "sha1-jpQ9E1jcN1VQVOy+LtsFqhdO3hQ="
789 | },
790 | "glob": {
791 | "version": "7.1.2",
792 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.2.tgz",
793 | "integrity": "sha512-MJTUg1kjuLeQCJ+ccE4Vpa6kKVXkPYJ2mOCQyUuKLcLQsdrMCpBPUi8qVE6+YuaJkozeA9NusTAw3hLr8Xe5EQ==",
794 | "requires": {
795 | "fs.realpath": "^1.0.0",
796 | "inflight": "^1.0.4",
797 | "inherits": "2",
798 | "minimatch": "^3.0.4",
799 | "once": "^1.3.0",
800 | "path-is-absolute": "^1.0.0"
801 | }
802 | },
803 | "globals": {
804 | "version": "11.5.0",
805 | "resolved": "https://registry.npmjs.org/globals/-/globals-11.5.0.tgz",
806 | "integrity": "sha512-hYyf+kI8dm3nORsiiXUQigOU62hDLfJ9G01uyGMxhc6BKsircrUhC4uJPQPUSuq2GrTmiiEt7ewxlMdBewfmKQ==",
807 | "dev": true
808 | },
809 | "globby": {
810 | "version": "5.0.0",
811 | "resolved": "https://registry.npmjs.org/globby/-/globby-5.0.0.tgz",
812 | "integrity": "sha1-69hGZ8oNuzMLmbz8aOrCvFQ3Dg0=",
813 | "dev": true,
814 | "requires": {
815 | "array-union": "^1.0.1",
816 | "arrify": "^1.0.0",
817 | "glob": "^7.0.3",
818 | "object-assign": "^4.0.1",
819 | "pify": "^2.0.0",
820 | "pinkie-promise": "^2.0.0"
821 | }
822 | },
823 | "graceful-fs": {
824 | "version": "4.1.11",
825 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.1.11.tgz",
826 | "integrity": "sha1-Dovf5NHduIVNZOBOp8AOKgJuVlg="
827 | },
828 | "has": {
829 | "version": "1.0.3",
830 | "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz",
831 | "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==",
832 | "dev": true,
833 | "requires": {
834 | "function-bind": "^1.1.1"
835 | }
836 | },
837 | "has-ansi": {
838 | "version": "2.0.0",
839 | "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz",
840 | "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=",
841 | "dev": true,
842 | "requires": {
843 | "ansi-regex": "^2.0.0"
844 | },
845 | "dependencies": {
846 | "ansi-regex": {
847 | "version": "2.1.1",
848 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz",
849 | "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=",
850 | "dev": true
851 | }
852 | }
853 | },
854 | "has-flag": {
855 | "version": "3.0.0",
856 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
857 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0="
858 | },
859 | "hoek": {
860 | "version": "5.0.3",
861 | "resolved": "https://registry.npmjs.org/hoek/-/hoek-5.0.3.tgz",
862 | "integrity": "sha512-Bmr56pxML1c9kU+NS51SMFkiVQAb+9uFfXwyqR2tn4w2FPvmPt65eZ9aCcEfRXd9G74HkZnILC6p967pED4aiw=="
863 | },
864 | "hosted-git-info": {
865 | "version": "2.6.0",
866 | "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.6.0.tgz",
867 | "integrity": "sha512-lIbgIIQA3lz5XaB6vxakj6sDHADJiZadYEJB+FgA+C4nubM1NwcuvUr9EJPmnH1skZqpqUzWborWo8EIUi0Sdw==",
868 | "dev": true
869 | },
870 | "https-proxy-agent": {
871 | "version": "2.2.1",
872 | "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-2.2.1.tgz",
873 | "integrity": "sha512-HPCTS1LW51bcyMYbxUIOO4HEOlQ1/1qRaFWcyxvwaqUS9TY88aoEuHUY33kuAh1YhVVaDQhLZsnPd+XNARWZlQ==",
874 | "requires": {
875 | "agent-base": "^4.1.0",
876 | "debug": "^3.1.0"
877 | }
878 | },
879 | "iconv-lite": {
880 | "version": "0.4.23",
881 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.23.tgz",
882 | "integrity": "sha512-neyTUVFtahjf0mB3dZT77u+8O0QB89jFdnBkd5P1JgYPbPaia3gXXOVL2fq8VyU2gMMD7SaN7QukTB/pmXYvDA==",
883 | "dev": true,
884 | "requires": {
885 | "safer-buffer": ">= 2.1.2 < 3"
886 | }
887 | },
888 | "ignore": {
889 | "version": "3.3.8",
890 | "resolved": "https://registry.npmjs.org/ignore/-/ignore-3.3.8.tgz",
891 | "integrity": "sha512-pUh+xUQQhQzevjRHHFqqcTy0/dP/kS9I8HSrUydhihjuD09W6ldVWFtIrwhXdUJHis3i2rZNqEHpZH/cbinFbg==",
892 | "dev": true
893 | },
894 | "imurmurhash": {
895 | "version": "0.1.4",
896 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz",
897 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=",
898 | "dev": true
899 | },
900 | "inflight": {
901 | "version": "1.0.6",
902 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz",
903 | "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=",
904 | "requires": {
905 | "once": "^1.3.0",
906 | "wrappy": "1"
907 | }
908 | },
909 | "inherits": {
910 | "version": "2.0.3",
911 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
912 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4="
913 | },
914 | "inquirer": {
915 | "version": "3.3.0",
916 | "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-3.3.0.tgz",
917 | "integrity": "sha512-h+xtnyk4EwKvFWHrUYsWErEVR+igKtLdchu+o0Z1RL7VU/jVMFbYir2bp6bAj8efFNxWqHX0dIss6fJQ+/+qeQ==",
918 | "dev": true,
919 | "requires": {
920 | "ansi-escapes": "^3.0.0",
921 | "chalk": "^2.0.0",
922 | "cli-cursor": "^2.1.0",
923 | "cli-width": "^2.0.0",
924 | "external-editor": "^2.0.4",
925 | "figures": "^2.0.0",
926 | "lodash": "^4.3.0",
927 | "mute-stream": "0.0.7",
928 | "run-async": "^2.2.0",
929 | "rx-lite": "^4.0.8",
930 | "rx-lite-aggregates": "^4.0.8",
931 | "string-width": "^2.1.0",
932 | "strip-ansi": "^4.0.0",
933 | "through": "^2.3.6"
934 | },
935 | "dependencies": {
936 | "ansi-escapes": {
937 | "version": "3.1.0",
938 | "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-3.1.0.tgz",
939 | "integrity": "sha512-UgAb8H9D41AQnu/PbWlCofQVcnV4Gs2bBJi9eZPxfU/hgglFh3SMDMENRIqdr7H6XFnXdoknctFByVsCOotTVw==",
940 | "dev": true
941 | }
942 | }
943 | },
944 | "invert-kv": {
945 | "version": "1.0.0",
946 | "resolved": "https://registry.npmjs.org/invert-kv/-/invert-kv-1.0.0.tgz",
947 | "integrity": "sha1-EEqOSqym09jNFXqO+L+rLXo//bY="
948 | },
949 | "is-arrayish": {
950 | "version": "0.2.1",
951 | "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz",
952 | "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=",
953 | "dev": true
954 | },
955 | "is-builtin-module": {
956 | "version": "1.0.0",
957 | "resolved": "https://registry.npmjs.org/is-builtin-module/-/is-builtin-module-1.0.0.tgz",
958 | "integrity": "sha1-VAVy0096wxGfj3bDDLwbHgN6/74=",
959 | "dev": true,
960 | "requires": {
961 | "builtin-modules": "^1.0.0"
962 | }
963 | },
964 | "is-fullwidth-code-point": {
965 | "version": "2.0.0",
966 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz",
967 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8="
968 | },
969 | "is-path-cwd": {
970 | "version": "1.0.0",
971 | "resolved": "https://registry.npmjs.org/is-path-cwd/-/is-path-cwd-1.0.0.tgz",
972 | "integrity": "sha1-0iXsIxMuie3Tj9p2dHLmLmXxEG0=",
973 | "dev": true
974 | },
975 | "is-path-in-cwd": {
976 | "version": "1.0.1",
977 | "resolved": "https://registry.npmjs.org/is-path-in-cwd/-/is-path-in-cwd-1.0.1.tgz",
978 | "integrity": "sha512-FjV1RTW48E7CWM7eE/J2NJvAEEVektecDBVBE5Hh3nM1Jd0kvhHtX68Pr3xsDf857xt3Y4AkwVULK1Vku62aaQ==",
979 | "dev": true,
980 | "requires": {
981 | "is-path-inside": "^1.0.0"
982 | }
983 | },
984 | "is-path-inside": {
985 | "version": "1.0.1",
986 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-1.0.1.tgz",
987 | "integrity": "sha1-jvW33lBDej/cprToZe96pVy0gDY=",
988 | "dev": true,
989 | "requires": {
990 | "path-is-inside": "^1.0.1"
991 | }
992 | },
993 | "is-promise": {
994 | "version": "2.1.0",
995 | "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-2.1.0.tgz",
996 | "integrity": "sha1-eaKp7OfwlugPNtKy87wWwf9L8/o=",
997 | "dev": true
998 | },
999 | "is-resolvable": {
1000 | "version": "1.1.0",
1001 | "resolved": "https://registry.npmjs.org/is-resolvable/-/is-resolvable-1.1.0.tgz",
1002 | "integrity": "sha512-qgDYXFSR5WvEfuS5dMj6oTMEbrrSaM0CrFk2Yiq/gXnBvD9pMa2jGXxyhGLfvhZpuMZe18CJpFxAt3CRs42NMg==",
1003 | "dev": true
1004 | },
1005 | "is-stream": {
1006 | "version": "1.1.0",
1007 | "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-1.1.0.tgz",
1008 | "integrity": "sha1-EtSj3U5o4Lec6428hBc66A2RykQ="
1009 | },
1010 | "isarray": {
1011 | "version": "1.0.0",
1012 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz",
1013 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE="
1014 | },
1015 | "isemail": {
1016 | "version": "3.1.2",
1017 | "resolved": "https://registry.npmjs.org/isemail/-/isemail-3.1.2.tgz",
1018 | "integrity": "sha512-zfRhJn9rFSGhzU5tGZqepRSAj3+g6oTOHxMGGriWNJZzyLPUK8H7VHpqKntegnW8KLyGA9zwuNaCoopl40LTpg==",
1019 | "requires": {
1020 | "punycode": "2.x.x"
1021 | }
1022 | },
1023 | "isexe": {
1024 | "version": "2.0.0",
1025 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz",
1026 | "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA="
1027 | },
1028 | "iterm2-version": {
1029 | "version": "2.3.0",
1030 | "resolved": "https://registry.npmjs.org/iterm2-version/-/iterm2-version-2.3.0.tgz",
1031 | "integrity": "sha1-rmQABGHgK18f5TMfC58Oxxzg4Tg=",
1032 | "requires": {
1033 | "app-path": "^2.1.0",
1034 | "plist": "^2.0.1"
1035 | }
1036 | },
1037 | "joi": {
1038 | "version": "13.4.0",
1039 | "resolved": "https://registry.npmjs.org/joi/-/joi-13.4.0.tgz",
1040 | "integrity": "sha512-JuK4GjEu6j7zr9FuVe2MAseZ6si/8/HaY0qMAejfDFHp7jcH4OKE937mIHM5VT4xDS0q7lpQbszbxKV9rm0yUg==",
1041 | "requires": {
1042 | "hoek": "5.x.x",
1043 | "isemail": "3.x.x",
1044 | "topo": "3.x.x"
1045 | }
1046 | },
1047 | "js-tokens": {
1048 | "version": "3.0.2",
1049 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-3.0.2.tgz",
1050 | "integrity": "sha1-mGbfOVECEw449/mWvOtlRDIJwls=",
1051 | "dev": true
1052 | },
1053 | "js-yaml": {
1054 | "version": "3.13.1",
1055 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.13.1.tgz",
1056 | "integrity": "sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==",
1057 | "dev": true,
1058 | "requires": {
1059 | "argparse": "^1.0.7",
1060 | "esprima": "^4.0.0"
1061 | }
1062 | },
1063 | "json-schema-traverse": {
1064 | "version": "0.3.1",
1065 | "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.3.1.tgz",
1066 | "integrity": "sha1-NJptRMU6Ud6JtAgFxdXlm0F9M0A=",
1067 | "dev": true
1068 | },
1069 | "json-stable-stringify-without-jsonify": {
1070 | "version": "1.0.1",
1071 | "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz",
1072 | "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=",
1073 | "dev": true
1074 | },
1075 | "jsonfile": {
1076 | "version": "4.0.0",
1077 | "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz",
1078 | "integrity": "sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss=",
1079 | "requires": {
1080 | "graceful-fs": "^4.1.6"
1081 | }
1082 | },
1083 | "lcid": {
1084 | "version": "1.0.0",
1085 | "resolved": "https://registry.npmjs.org/lcid/-/lcid-1.0.0.tgz",
1086 | "integrity": "sha1-MIrMr6C8SDo4Z7S28rlQYlHRuDU=",
1087 | "requires": {
1088 | "invert-kv": "^1.0.0"
1089 | }
1090 | },
1091 | "levn": {
1092 | "version": "0.3.0",
1093 | "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz",
1094 | "integrity": "sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4=",
1095 | "dev": true,
1096 | "requires": {
1097 | "prelude-ls": "~1.1.2",
1098 | "type-check": "~0.3.2"
1099 | }
1100 | },
1101 | "load-json-file": {
1102 | "version": "2.0.0",
1103 | "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-2.0.0.tgz",
1104 | "integrity": "sha1-eUfkIUmvgNaWy/eXvKq8/h/inKg=",
1105 | "dev": true,
1106 | "requires": {
1107 | "graceful-fs": "^4.1.2",
1108 | "parse-json": "^2.2.0",
1109 | "pify": "^2.0.0",
1110 | "strip-bom": "^3.0.0"
1111 | }
1112 | },
1113 | "locate-path": {
1114 | "version": "2.0.0",
1115 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz",
1116 | "integrity": "sha1-K1aLJl7slExtnA3pw9u7ygNUzY4=",
1117 | "requires": {
1118 | "p-locate": "^2.0.0",
1119 | "path-exists": "^3.0.0"
1120 | }
1121 | },
1122 | "lodash": {
1123 | "version": "4.17.15",
1124 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.15.tgz",
1125 | "integrity": "sha512-8xOcRHvCjnocdS5cpwXQXVzmmh5e5+saE2QGoeQmbKmRS6J3VQppPOIt0MnmE+4xlZoumy0GPG0D0MVIQbNA1A==",
1126 | "dev": true
1127 | },
1128 | "log-symbols": {
1129 | "version": "2.2.0",
1130 | "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-2.2.0.tgz",
1131 | "integrity": "sha512-VeIAFslyIerEJLXHziedo2basKbMKtTw3vfn5IzG0XTjhAVEJyNHnL2p7vc+wBDSdQuUpNw3M2u6xb9QsAY5Eg==",
1132 | "requires": {
1133 | "chalk": "^2.0.1"
1134 | }
1135 | },
1136 | "lru-cache": {
1137 | "version": "4.1.3",
1138 | "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-4.1.3.tgz",
1139 | "integrity": "sha512-fFEhvcgzuIoJVUF8fYr5KR0YqxD238zgObTps31YdADwPPAp82a4M8TrckkWyx7ekNlf9aBcVn81cFwwXngrJA==",
1140 | "requires": {
1141 | "pseudomap": "^1.0.2",
1142 | "yallist": "^2.1.2"
1143 | }
1144 | },
1145 | "mem": {
1146 | "version": "1.1.0",
1147 | "resolved": "https://registry.npmjs.org/mem/-/mem-1.1.0.tgz",
1148 | "integrity": "sha1-Xt1StIXKHZAP5kiVUFOZoN+kX3Y=",
1149 | "requires": {
1150 | "mimic-fn": "^1.0.0"
1151 | }
1152 | },
1153 | "mime": {
1154 | "version": "2.3.1",
1155 | "resolved": "https://registry.npmjs.org/mime/-/mime-2.3.1.tgz",
1156 | "integrity": "sha512-OEUllcVoydBHGN1z84yfQDimn58pZNNNXgZlHXSboxMlFvgI6MXSWpWKpFRra7H1HxpVhHTkrghfRW49k6yjeg=="
1157 | },
1158 | "mimic-fn": {
1159 | "version": "1.2.0",
1160 | "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-1.2.0.tgz",
1161 | "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ=="
1162 | },
1163 | "minimatch": {
1164 | "version": "3.0.4",
1165 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz",
1166 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
1167 | "requires": {
1168 | "brace-expansion": "^1.1.7"
1169 | }
1170 | },
1171 | "minimist": {
1172 | "version": "0.0.8",
1173 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz",
1174 | "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0="
1175 | },
1176 | "mkdirp": {
1177 | "version": "0.5.1",
1178 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz",
1179 | "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=",
1180 | "requires": {
1181 | "minimist": "0.0.8"
1182 | }
1183 | },
1184 | "ms": {
1185 | "version": "2.0.0",
1186 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
1187 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
1188 | },
1189 | "mute-stream": {
1190 | "version": "0.0.7",
1191 | "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.7.tgz",
1192 | "integrity": "sha1-MHXOk7whuPq0PhvE2n6BFe0ee6s=",
1193 | "dev": true
1194 | },
1195 | "natural-compare": {
1196 | "version": "1.4.0",
1197 | "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz",
1198 | "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=",
1199 | "dev": true
1200 | },
1201 | "normalize-package-data": {
1202 | "version": "2.4.0",
1203 | "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.4.0.tgz",
1204 | "integrity": "sha512-9jjUFbTPfEy3R/ad/2oNbKtW9Hgovl5O1FvFWKkKblNXoN/Oou6+9+KKohPK13Yc3/TyunyWhJp6gvRNR/PPAw==",
1205 | "dev": true,
1206 | "requires": {
1207 | "hosted-git-info": "^2.1.4",
1208 | "is-builtin-module": "^1.0.0",
1209 | "semver": "2 || 3 || 4 || 5",
1210 | "validate-npm-package-license": "^3.0.1"
1211 | }
1212 | },
1213 | "npm-run-path": {
1214 | "version": "1.0.0",
1215 | "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-1.0.0.tgz",
1216 | "integrity": "sha1-9cMr9ZX+ga6Sfa7FLoL4sACsPI8=",
1217 | "requires": {
1218 | "path-key": "^1.0.0"
1219 | }
1220 | },
1221 | "number-is-nan": {
1222 | "version": "1.0.1",
1223 | "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz",
1224 | "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0="
1225 | },
1226 | "object-assign": {
1227 | "version": "4.1.1",
1228 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
1229 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM="
1230 | },
1231 | "once": {
1232 | "version": "1.4.0",
1233 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
1234 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=",
1235 | "requires": {
1236 | "wrappy": "1"
1237 | }
1238 | },
1239 | "onetime": {
1240 | "version": "2.0.1",
1241 | "resolved": "https://registry.npmjs.org/onetime/-/onetime-2.0.1.tgz",
1242 | "integrity": "sha1-BnQoIw/WdEOyeUsiu6UotoZ5YtQ=",
1243 | "requires": {
1244 | "mimic-fn": "^1.0.0"
1245 | }
1246 | },
1247 | "optionator": {
1248 | "version": "0.8.2",
1249 | "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.2.tgz",
1250 | "integrity": "sha1-NkxeQJ0/TWMB1sC0wFu6UBgK62Q=",
1251 | "dev": true,
1252 | "requires": {
1253 | "deep-is": "~0.1.3",
1254 | "fast-levenshtein": "~2.0.4",
1255 | "levn": "~0.3.0",
1256 | "prelude-ls": "~1.1.2",
1257 | "type-check": "~0.3.2",
1258 | "wordwrap": "~1.0.0"
1259 | }
1260 | },
1261 | "ora": {
1262 | "version": "2.1.0",
1263 | "resolved": "https://registry.npmjs.org/ora/-/ora-2.1.0.tgz",
1264 | "integrity": "sha512-hNNlAd3gfv/iPmsNxYoAPLvxg7HuPozww7fFonMZvL84tP6Ox5igfk5j/+a9rtJJwqMgKK+JgWsAQik5o0HTLA==",
1265 | "requires": {
1266 | "chalk": "^2.3.1",
1267 | "cli-cursor": "^2.1.0",
1268 | "cli-spinners": "^1.1.0",
1269 | "log-symbols": "^2.2.0",
1270 | "strip-ansi": "^4.0.0",
1271 | "wcwidth": "^1.0.1"
1272 | }
1273 | },
1274 | "os-locale": {
1275 | "version": "2.1.0",
1276 | "resolved": "https://registry.npmjs.org/os-locale/-/os-locale-2.1.0.tgz",
1277 | "integrity": "sha512-3sslG3zJbEYcaC4YVAvDorjGxc7tv6KVATnLPZONiljsUncvihe9BQoVCEs0RZ1kmf4Hk9OBqlZfJZWI4GanKA==",
1278 | "requires": {
1279 | "execa": "^0.7.0",
1280 | "lcid": "^1.0.0",
1281 | "mem": "^1.1.0"
1282 | },
1283 | "dependencies": {
1284 | "execa": {
1285 | "version": "0.7.0",
1286 | "resolved": "https://registry.npmjs.org/execa/-/execa-0.7.0.tgz",
1287 | "integrity": "sha1-lEvs00zEHuMqY6n68nrVpl/Fl3c=",
1288 | "requires": {
1289 | "cross-spawn": "^5.0.1",
1290 | "get-stream": "^3.0.0",
1291 | "is-stream": "^1.1.0",
1292 | "npm-run-path": "^2.0.0",
1293 | "p-finally": "^1.0.0",
1294 | "signal-exit": "^3.0.0",
1295 | "strip-eof": "^1.0.0"
1296 | }
1297 | },
1298 | "npm-run-path": {
1299 | "version": "2.0.2",
1300 | "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-2.0.2.tgz",
1301 | "integrity": "sha1-NakjLfo11wZ7TLLd8jV7GHFTbF8=",
1302 | "requires": {
1303 | "path-key": "^2.0.0"
1304 | }
1305 | },
1306 | "path-key": {
1307 | "version": "2.0.1",
1308 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz",
1309 | "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A="
1310 | }
1311 | }
1312 | },
1313 | "os-tmpdir": {
1314 | "version": "1.0.2",
1315 | "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz",
1316 | "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=",
1317 | "dev": true
1318 | },
1319 | "p-finally": {
1320 | "version": "1.0.0",
1321 | "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz",
1322 | "integrity": "sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4="
1323 | },
1324 | "p-limit": {
1325 | "version": "1.3.0",
1326 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-1.3.0.tgz",
1327 | "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==",
1328 | "requires": {
1329 | "p-try": "^1.0.0"
1330 | }
1331 | },
1332 | "p-locate": {
1333 | "version": "2.0.0",
1334 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz",
1335 | "integrity": "sha1-IKAQOyIqcMj9OcwuWAaA893l7EM=",
1336 | "requires": {
1337 | "p-limit": "^1.1.0"
1338 | }
1339 | },
1340 | "p-try": {
1341 | "version": "1.0.0",
1342 | "resolved": "https://registry.npmjs.org/p-try/-/p-try-1.0.0.tgz",
1343 | "integrity": "sha1-y8ec26+P1CKOE/Yh8rGiN8GyB7M="
1344 | },
1345 | "parse-json": {
1346 | "version": "2.2.0",
1347 | "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-2.2.0.tgz",
1348 | "integrity": "sha1-9ID0BDTvgHQfhGkJn43qGPVaTck=",
1349 | "dev": true,
1350 | "requires": {
1351 | "error-ex": "^1.2.0"
1352 | }
1353 | },
1354 | "path-exists": {
1355 | "version": "3.0.0",
1356 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz",
1357 | "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU="
1358 | },
1359 | "path-is-absolute": {
1360 | "version": "1.0.1",
1361 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz",
1362 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18="
1363 | },
1364 | "path-is-inside": {
1365 | "version": "1.0.2",
1366 | "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz",
1367 | "integrity": "sha1-NlQX3t5EQw0cEa9hAn+s8HS9/FM=",
1368 | "dev": true
1369 | },
1370 | "path-key": {
1371 | "version": "1.0.0",
1372 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-1.0.0.tgz",
1373 | "integrity": "sha1-XVPVeAGWRsDWiADbThRua9wqx68="
1374 | },
1375 | "path-parse": {
1376 | "version": "1.0.5",
1377 | "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.5.tgz",
1378 | "integrity": "sha1-PBrfhx6pzWyUMbbqK9dKD/BVxME=",
1379 | "dev": true
1380 | },
1381 | "path-type": {
1382 | "version": "2.0.0",
1383 | "resolved": "https://registry.npmjs.org/path-type/-/path-type-2.0.0.tgz",
1384 | "integrity": "sha1-8BLMuEFbcJb8LaoQVMPXI4lZTHM=",
1385 | "dev": true,
1386 | "requires": {
1387 | "pify": "^2.0.0"
1388 | }
1389 | },
1390 | "pend": {
1391 | "version": "1.2.0",
1392 | "resolved": "https://registry.npmjs.org/pend/-/pend-1.2.0.tgz",
1393 | "integrity": "sha1-elfrVQpng/kRUzH89GY9XI4AelA="
1394 | },
1395 | "pify": {
1396 | "version": "2.3.0",
1397 | "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz",
1398 | "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=",
1399 | "dev": true
1400 | },
1401 | "pinkie": {
1402 | "version": "2.0.4",
1403 | "resolved": "https://registry.npmjs.org/pinkie/-/pinkie-2.0.4.tgz",
1404 | "integrity": "sha1-clVrgM+g1IqXToDnckjoDtT3+HA=",
1405 | "dev": true
1406 | },
1407 | "pinkie-promise": {
1408 | "version": "2.0.1",
1409 | "resolved": "https://registry.npmjs.org/pinkie-promise/-/pinkie-promise-2.0.1.tgz",
1410 | "integrity": "sha1-ITXW36ejWMBprJsXh3YogihFD/o=",
1411 | "dev": true,
1412 | "requires": {
1413 | "pinkie": "^2.0.0"
1414 | }
1415 | },
1416 | "pixelmatch": {
1417 | "version": "4.0.2",
1418 | "resolved": "https://registry.npmjs.org/pixelmatch/-/pixelmatch-4.0.2.tgz",
1419 | "integrity": "sha1-j0fc7FARtHe2fbA8JDvB8wheiFQ=",
1420 | "requires": {
1421 | "pngjs": "^3.0.0"
1422 | }
1423 | },
1424 | "pkg-dir": {
1425 | "version": "1.0.0",
1426 | "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-1.0.0.tgz",
1427 | "integrity": "sha1-ektQio1bstYp1EcFb/TpyTFM89Q=",
1428 | "dev": true,
1429 | "requires": {
1430 | "find-up": "^1.0.0"
1431 | },
1432 | "dependencies": {
1433 | "find-up": {
1434 | "version": "1.1.2",
1435 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-1.1.2.tgz",
1436 | "integrity": "sha1-ay6YIrGizgpgq2TWEOzK1TyyTQ8=",
1437 | "dev": true,
1438 | "requires": {
1439 | "path-exists": "^2.0.0",
1440 | "pinkie-promise": "^2.0.0"
1441 | }
1442 | },
1443 | "path-exists": {
1444 | "version": "2.1.0",
1445 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-2.1.0.tgz",
1446 | "integrity": "sha1-D+tsZPD8UY2adU3V77YscCJ2H0s=",
1447 | "dev": true,
1448 | "requires": {
1449 | "pinkie-promise": "^2.0.0"
1450 | }
1451 | }
1452 | }
1453 | },
1454 | "plist": {
1455 | "version": "2.1.0",
1456 | "resolved": "https://registry.npmjs.org/plist/-/plist-2.1.0.tgz",
1457 | "integrity": "sha1-V8zbeggh3yGDEhejytVOPhRqECU=",
1458 | "requires": {
1459 | "base64-js": "1.2.0",
1460 | "xmlbuilder": "8.2.2",
1461 | "xmldom": "0.1.x"
1462 | }
1463 | },
1464 | "pluralize": {
1465 | "version": "7.0.0",
1466 | "resolved": "https://registry.npmjs.org/pluralize/-/pluralize-7.0.0.tgz",
1467 | "integrity": "sha512-ARhBOdzS3e41FbkW/XWrTEtukqqLoK5+Z/4UeDaLuSW+39JPeFgs4gCGqsrJHVZX0fUrx//4OF0K1CUGwlIFow==",
1468 | "dev": true
1469 | },
1470 | "pngjs": {
1471 | "version": "3.3.3",
1472 | "resolved": "https://registry.npmjs.org/pngjs/-/pngjs-3.3.3.tgz",
1473 | "integrity": "sha512-1n3Z4p3IOxArEs1VRXnZ/RXdfEniAUS9jb68g58FIXMNkPJeZd+Qh4Uq7/e0LVxAQGos1eIUrqrt4FpjdnEd+Q=="
1474 | },
1475 | "prelude-ls": {
1476 | "version": "1.1.2",
1477 | "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz",
1478 | "integrity": "sha1-IZMqVJ9eUv/ZqCf1cOBL5iqX2lQ=",
1479 | "dev": true
1480 | },
1481 | "process-nextick-args": {
1482 | "version": "2.0.0",
1483 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.0.tgz",
1484 | "integrity": "sha512-MtEC1TqN0EU5nephaJ4rAtThHtC86dNN9qCuEhtshvpVBkAW5ZO7BASN9REnF9eoXGcRub+pFuKEpOHE+HbEMw=="
1485 | },
1486 | "progress": {
1487 | "version": "2.0.0",
1488 | "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.0.tgz",
1489 | "integrity": "sha1-ihvjZr+Pwj2yvSPxDG/pILQ4nR8="
1490 | },
1491 | "proxy-from-env": {
1492 | "version": "1.0.0",
1493 | "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.0.0.tgz",
1494 | "integrity": "sha1-M8UDmPcOp+uW0h97gXYwpVeRx+4="
1495 | },
1496 | "pseudomap": {
1497 | "version": "1.0.2",
1498 | "resolved": "https://registry.npmjs.org/pseudomap/-/pseudomap-1.0.2.tgz",
1499 | "integrity": "sha1-8FKijacOYYkX7wqKw0wa5aaChrM="
1500 | },
1501 | "punycode": {
1502 | "version": "2.1.1",
1503 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz",
1504 | "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A=="
1505 | },
1506 | "puppeteer": {
1507 | "version": "1.5.0",
1508 | "resolved": "https://registry.npmjs.org/puppeteer/-/puppeteer-1.5.0.tgz",
1509 | "integrity": "sha512-eELwFtFxL+uhmg4jPZOZXzSrPEYy4CaYQNbcchBbfxY+KjMpnv6XGf/aYWaQG49OTpfi2/DMziXtDM8XuJgoUA==",
1510 | "requires": {
1511 | "debug": "^3.1.0",
1512 | "extract-zip": "^1.6.6",
1513 | "https-proxy-agent": "^2.2.1",
1514 | "mime": "^2.0.3",
1515 | "progress": "^2.0.0",
1516 | "proxy-from-env": "^1.0.0",
1517 | "rimraf": "^2.6.1",
1518 | "ws": "^5.1.1"
1519 | }
1520 | },
1521 | "read-pkg": {
1522 | "version": "2.0.0",
1523 | "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-2.0.0.tgz",
1524 | "integrity": "sha1-jvHAYjxqbbDcZxPEv6xGMysjaPg=",
1525 | "dev": true,
1526 | "requires": {
1527 | "load-json-file": "^2.0.0",
1528 | "normalize-package-data": "^2.3.2",
1529 | "path-type": "^2.0.0"
1530 | }
1531 | },
1532 | "read-pkg-up": {
1533 | "version": "2.0.0",
1534 | "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-2.0.0.tgz",
1535 | "integrity": "sha1-a3KoBImE4MQeeVEP1en6mbO1Sb4=",
1536 | "dev": true,
1537 | "requires": {
1538 | "find-up": "^2.0.0",
1539 | "read-pkg": "^2.0.0"
1540 | }
1541 | },
1542 | "readable-stream": {
1543 | "version": "2.3.6",
1544 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.6.tgz",
1545 | "integrity": "sha512-tQtKA9WIAhBF3+VLAseyMqZeBjW0AHJoxOtYqSUZNJxauErmLbVm2FW1y+J/YA9dUrAC39ITejlZWhVIwawkKw==",
1546 | "requires": {
1547 | "core-util-is": "~1.0.0",
1548 | "inherits": "~2.0.3",
1549 | "isarray": "~1.0.0",
1550 | "process-nextick-args": "~2.0.0",
1551 | "safe-buffer": "~5.1.1",
1552 | "string_decoder": "~1.1.1",
1553 | "util-deprecate": "~1.0.1"
1554 | }
1555 | },
1556 | "regexpp": {
1557 | "version": "1.1.0",
1558 | "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-1.1.0.tgz",
1559 | "integrity": "sha512-LOPw8FpgdQF9etWMaAfG/WRthIdXJGYp4mJ2Jgn/2lpkbod9jPn0t9UqN7AxBOKNfzRbYyVfgc7Vk4t/MpnXgw==",
1560 | "dev": true
1561 | },
1562 | "require-directory": {
1563 | "version": "2.1.1",
1564 | "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz",
1565 | "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I="
1566 | },
1567 | "require-main-filename": {
1568 | "version": "1.0.1",
1569 | "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-1.0.1.tgz",
1570 | "integrity": "sha1-l/cXtp1IeE9fUmpsWqj/3aBVpNE="
1571 | },
1572 | "require-uncached": {
1573 | "version": "1.0.3",
1574 | "resolved": "https://registry.npmjs.org/require-uncached/-/require-uncached-1.0.3.tgz",
1575 | "integrity": "sha1-Tg1W1slmL9MeQwEcS5WqSZVUIdM=",
1576 | "dev": true,
1577 | "requires": {
1578 | "caller-path": "^0.1.0",
1579 | "resolve-from": "^1.0.0"
1580 | }
1581 | },
1582 | "resolve": {
1583 | "version": "1.7.1",
1584 | "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.7.1.tgz",
1585 | "integrity": "sha512-c7rwLofp8g1U+h1KNyHL/jicrKg1Ek4q+Lr33AL65uZTinUZHe30D5HlyN5V9NW0JX1D5dXQ4jqW5l7Sy/kGfw==",
1586 | "dev": true,
1587 | "requires": {
1588 | "path-parse": "^1.0.5"
1589 | }
1590 | },
1591 | "resolve-from": {
1592 | "version": "1.0.1",
1593 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-1.0.1.tgz",
1594 | "integrity": "sha1-Jsv+k10a7uq7Kbw/5a6wHpPUQiY=",
1595 | "dev": true
1596 | },
1597 | "restore-cursor": {
1598 | "version": "2.0.0",
1599 | "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-2.0.0.tgz",
1600 | "integrity": "sha1-n37ih/gv0ybU/RYpI9YhKe7g368=",
1601 | "requires": {
1602 | "onetime": "^2.0.0",
1603 | "signal-exit": "^3.0.2"
1604 | }
1605 | },
1606 | "rimraf": {
1607 | "version": "2.6.2",
1608 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.2.tgz",
1609 | "integrity": "sha512-lreewLK/BlghmxtfH36YYVg1i8IAce4TI7oao75I1g245+6BctqTVQiBP3YUJ9C6DQOXJmkYR9X9fCLtCOJc5w==",
1610 | "requires": {
1611 | "glob": "^7.0.5"
1612 | }
1613 | },
1614 | "run-async": {
1615 | "version": "2.3.0",
1616 | "resolved": "https://registry.npmjs.org/run-async/-/run-async-2.3.0.tgz",
1617 | "integrity": "sha1-A3GrSuC91yDUFm19/aZP96RFpsA=",
1618 | "dev": true,
1619 | "requires": {
1620 | "is-promise": "^2.1.0"
1621 | }
1622 | },
1623 | "rx-lite": {
1624 | "version": "4.0.8",
1625 | "resolved": "https://registry.npmjs.org/rx-lite/-/rx-lite-4.0.8.tgz",
1626 | "integrity": "sha1-Cx4Rr4vESDbwSmQH6S2kJGe3lEQ=",
1627 | "dev": true
1628 | },
1629 | "rx-lite-aggregates": {
1630 | "version": "4.0.8",
1631 | "resolved": "https://registry.npmjs.org/rx-lite-aggregates/-/rx-lite-aggregates-4.0.8.tgz",
1632 | "integrity": "sha1-dTuHqJoRyVRnxKwWJsTvxOBcZ74=",
1633 | "dev": true,
1634 | "requires": {
1635 | "rx-lite": "*"
1636 | }
1637 | },
1638 | "safe-buffer": {
1639 | "version": "5.1.2",
1640 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
1641 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
1642 | },
1643 | "safer-buffer": {
1644 | "version": "2.1.2",
1645 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
1646 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==",
1647 | "dev": true
1648 | },
1649 | "semver": {
1650 | "version": "5.5.0",
1651 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.5.0.tgz",
1652 | "integrity": "sha512-4SJ3dm0WAwWy/NVeioZh5AntkdJoWKxHxcmyP622fOkgHa4z3R0TdBJICINyaSDE6uNwVc8gZr+ZinwZAH4xIA==",
1653 | "dev": true
1654 | },
1655 | "set-blocking": {
1656 | "version": "2.0.0",
1657 | "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz",
1658 | "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc="
1659 | },
1660 | "shebang-command": {
1661 | "version": "1.2.0",
1662 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz",
1663 | "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=",
1664 | "requires": {
1665 | "shebang-regex": "^1.0.0"
1666 | }
1667 | },
1668 | "shebang-regex": {
1669 | "version": "1.0.0",
1670 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz",
1671 | "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM="
1672 | },
1673 | "signal-exit": {
1674 | "version": "3.0.2",
1675 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.2.tgz",
1676 | "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0="
1677 | },
1678 | "slice-ansi": {
1679 | "version": "1.0.0",
1680 | "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-1.0.0.tgz",
1681 | "integrity": "sha512-POqxBK6Lb3q6s047D/XsDVNPnF9Dl8JSaqe9h9lURl0OdNqy/ujDrOiIHtsqXMGbWWTIomRzAMaTyawAU//Reg==",
1682 | "dev": true,
1683 | "requires": {
1684 | "is-fullwidth-code-point": "^2.0.0"
1685 | }
1686 | },
1687 | "spdx-correct": {
1688 | "version": "3.0.0",
1689 | "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.0.0.tgz",
1690 | "integrity": "sha512-N19o9z5cEyc8yQQPukRCZ9EUmb4HUpnrmaL/fxS2pBo2jbfcFRVuFZ/oFC+vZz0MNNk0h80iMn5/S6qGZOL5+g==",
1691 | "dev": true,
1692 | "requires": {
1693 | "spdx-expression-parse": "^3.0.0",
1694 | "spdx-license-ids": "^3.0.0"
1695 | }
1696 | },
1697 | "spdx-exceptions": {
1698 | "version": "2.1.0",
1699 | "resolved": "https://registry.npmjs.org/spdx-exceptions/-/spdx-exceptions-2.1.0.tgz",
1700 | "integrity": "sha512-4K1NsmrlCU1JJgUrtgEeTVyfx8VaYea9J9LvARxhbHtVtohPs/gFGG5yy49beySjlIMhhXZ4QqujIZEfS4l6Cg==",
1701 | "dev": true
1702 | },
1703 | "spdx-expression-parse": {
1704 | "version": "3.0.0",
1705 | "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-3.0.0.tgz",
1706 | "integrity": "sha512-Yg6D3XpRD4kkOmTpdgbUiEJFKghJH03fiC1OPll5h/0sO6neh2jqRDVHOQ4o/LMea0tgCkbMgea5ip/e+MkWyg==",
1707 | "dev": true,
1708 | "requires": {
1709 | "spdx-exceptions": "^2.1.0",
1710 | "spdx-license-ids": "^3.0.0"
1711 | }
1712 | },
1713 | "spdx-license-ids": {
1714 | "version": "3.0.0",
1715 | "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.0.tgz",
1716 | "integrity": "sha512-2+EPwgbnmOIl8HjGBXXMd9NAu02vLjOO1nWw4kmeRDFyHn+M/ETfHxQUK0oXg8ctgVnl9t3rosNVsZ1jG61nDA==",
1717 | "dev": true
1718 | },
1719 | "sprintf-js": {
1720 | "version": "1.0.3",
1721 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz",
1722 | "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=",
1723 | "dev": true
1724 | },
1725 | "string-width": {
1726 | "version": "2.1.1",
1727 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz",
1728 | "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==",
1729 | "requires": {
1730 | "is-fullwidth-code-point": "^2.0.0",
1731 | "strip-ansi": "^4.0.0"
1732 | }
1733 | },
1734 | "string_decoder": {
1735 | "version": "1.1.1",
1736 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz",
1737 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==",
1738 | "requires": {
1739 | "safe-buffer": "~5.1.0"
1740 | }
1741 | },
1742 | "strip-ansi": {
1743 | "version": "4.0.0",
1744 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz",
1745 | "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=",
1746 | "requires": {
1747 | "ansi-regex": "^3.0.0"
1748 | }
1749 | },
1750 | "strip-bom": {
1751 | "version": "3.0.0",
1752 | "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz",
1753 | "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=",
1754 | "dev": true
1755 | },
1756 | "strip-eof": {
1757 | "version": "1.0.0",
1758 | "resolved": "https://registry.npmjs.org/strip-eof/-/strip-eof-1.0.0.tgz",
1759 | "integrity": "sha1-u0P/VZim6wXYm1n80SnJgzE2Br8="
1760 | },
1761 | "strip-json-comments": {
1762 | "version": "2.0.1",
1763 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz",
1764 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=",
1765 | "dev": true
1766 | },
1767 | "supports-color": {
1768 | "version": "5.4.0",
1769 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.4.0.tgz",
1770 | "integrity": "sha512-zjaXglF5nnWpsq470jSv6P9DwPvgLkuapYmfDm3JWOm0vkNTVF2tI4UrN2r6jH1qM/uc/WtxYY1hYoA2dOKj5w==",
1771 | "requires": {
1772 | "has-flag": "^3.0.0"
1773 | }
1774 | },
1775 | "table": {
1776 | "version": "4.0.2",
1777 | "resolved": "https://registry.npmjs.org/table/-/table-4.0.2.tgz",
1778 | "integrity": "sha512-UUkEAPdSGxtRpiV9ozJ5cMTtYiqz7Ni1OGqLXRCynrvzdtR1p+cfOWe2RJLwvUG8hNanaSRjecIqwOjqeatDsA==",
1779 | "dev": true,
1780 | "requires": {
1781 | "ajv": "^5.2.3",
1782 | "ajv-keywords": "^2.1.0",
1783 | "chalk": "^2.1.0",
1784 | "lodash": "^4.17.4",
1785 | "slice-ansi": "1.0.0",
1786 | "string-width": "^2.1.1"
1787 | }
1788 | },
1789 | "term-img": {
1790 | "version": "2.1.0",
1791 | "resolved": "https://registry.npmjs.org/term-img/-/term-img-2.1.0.tgz",
1792 | "integrity": "sha512-j78Y+26QYTTWvtVVCmDx94idvQm6p59E+xRfQDSevIyM8dg45uUAtr/xbu13l0BeKrebPyUpgh8PM3noXlIBkw==",
1793 | "requires": {
1794 | "ansi-escapes": "^2.0.0",
1795 | "iterm2-version": "^2.1.0"
1796 | }
1797 | },
1798 | "text-table": {
1799 | "version": "0.2.0",
1800 | "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz",
1801 | "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=",
1802 | "dev": true
1803 | },
1804 | "through": {
1805 | "version": "2.3.8",
1806 | "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz",
1807 | "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=",
1808 | "dev": true
1809 | },
1810 | "tmp": {
1811 | "version": "0.0.33",
1812 | "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz",
1813 | "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==",
1814 | "dev": true,
1815 | "requires": {
1816 | "os-tmpdir": "~1.0.2"
1817 | }
1818 | },
1819 | "topo": {
1820 | "version": "3.0.0",
1821 | "resolved": "https://registry.npmjs.org/topo/-/topo-3.0.0.tgz",
1822 | "integrity": "sha512-Tlu1fGlR90iCdIPURqPiufqAlCZYzLjHYVVbcFWDMcX7+tK8hdZWAfsMrD/pBul9jqHHwFjNdf1WaxA9vTRRhw==",
1823 | "requires": {
1824 | "hoek": "5.x.x"
1825 | }
1826 | },
1827 | "type-check": {
1828 | "version": "0.3.2",
1829 | "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz",
1830 | "integrity": "sha1-WITKtRLPHTVeP7eE8wgEsrUg23I=",
1831 | "dev": true,
1832 | "requires": {
1833 | "prelude-ls": "~1.1.2"
1834 | }
1835 | },
1836 | "typedarray": {
1837 | "version": "0.0.6",
1838 | "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz",
1839 | "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c="
1840 | },
1841 | "universalify": {
1842 | "version": "0.1.1",
1843 | "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.1.tgz",
1844 | "integrity": "sha1-+nG63UQ3r0wUiEHjs7Fl+enlkLc="
1845 | },
1846 | "util-deprecate": {
1847 | "version": "1.0.2",
1848 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
1849 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8="
1850 | },
1851 | "validate-npm-package-license": {
1852 | "version": "3.0.3",
1853 | "resolved": "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.3.tgz",
1854 | "integrity": "sha512-63ZOUnL4SIXj4L0NixR3L1lcjO38crAbgrTpl28t8jjrfuiOBL5Iygm+60qPs/KsZGzPNg6Smnc/oY16QTjF0g==",
1855 | "dev": true,
1856 | "requires": {
1857 | "spdx-correct": "^3.0.0",
1858 | "spdx-expression-parse": "^3.0.0"
1859 | }
1860 | },
1861 | "wcwidth": {
1862 | "version": "1.0.1",
1863 | "resolved": "https://registry.npmjs.org/wcwidth/-/wcwidth-1.0.1.tgz",
1864 | "integrity": "sha1-8LDc+RW8X/FSivrbLA4XtTLaL+g=",
1865 | "requires": {
1866 | "defaults": "^1.0.3"
1867 | }
1868 | },
1869 | "which": {
1870 | "version": "1.3.1",
1871 | "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz",
1872 | "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==",
1873 | "requires": {
1874 | "isexe": "^2.0.0"
1875 | }
1876 | },
1877 | "which-module": {
1878 | "version": "2.0.0",
1879 | "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz",
1880 | "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho="
1881 | },
1882 | "wordwrap": {
1883 | "version": "1.0.0",
1884 | "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz",
1885 | "integrity": "sha1-J1hIEIkUVqQXHI0CJkQa3pDLyus=",
1886 | "dev": true
1887 | },
1888 | "wrap-ansi": {
1889 | "version": "2.1.0",
1890 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-2.1.0.tgz",
1891 | "integrity": "sha1-2Pw9KE3QV5T+hJc8rs3Rz4JP3YU=",
1892 | "requires": {
1893 | "string-width": "^1.0.1",
1894 | "strip-ansi": "^3.0.1"
1895 | },
1896 | "dependencies": {
1897 | "ansi-regex": {
1898 | "version": "2.1.1",
1899 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz",
1900 | "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8="
1901 | },
1902 | "is-fullwidth-code-point": {
1903 | "version": "1.0.0",
1904 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz",
1905 | "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=",
1906 | "requires": {
1907 | "number-is-nan": "^1.0.0"
1908 | }
1909 | },
1910 | "string-width": {
1911 | "version": "1.0.2",
1912 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz",
1913 | "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=",
1914 | "requires": {
1915 | "code-point-at": "^1.0.0",
1916 | "is-fullwidth-code-point": "^1.0.0",
1917 | "strip-ansi": "^3.0.0"
1918 | }
1919 | },
1920 | "strip-ansi": {
1921 | "version": "3.0.1",
1922 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz",
1923 | "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=",
1924 | "requires": {
1925 | "ansi-regex": "^2.0.0"
1926 | }
1927 | }
1928 | }
1929 | },
1930 | "wrappy": {
1931 | "version": "1.0.2",
1932 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
1933 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8="
1934 | },
1935 | "write": {
1936 | "version": "0.2.1",
1937 | "resolved": "https://registry.npmjs.org/write/-/write-0.2.1.tgz",
1938 | "integrity": "sha1-X8A4KOJkzqP+kUVUdvejxWbLB1c=",
1939 | "dev": true,
1940 | "requires": {
1941 | "mkdirp": "^0.5.1"
1942 | }
1943 | },
1944 | "ws": {
1945 | "version": "5.2.0",
1946 | "resolved": "https://registry.npmjs.org/ws/-/ws-5.2.0.tgz",
1947 | "integrity": "sha512-c18dMeW+PEQdDFzkhDsnBAlS4Z8KGStBQQUcQ5mf7Nf689jyGk0594L+i9RaQuf4gog6SvWLJorz2NfSaqxZ7w==",
1948 | "requires": {
1949 | "async-limiter": "~1.0.0"
1950 | }
1951 | },
1952 | "xmlbuilder": {
1953 | "version": "8.2.2",
1954 | "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-8.2.2.tgz",
1955 | "integrity": "sha1-aSSGc0ELS6QuGmE2VR0pIjNap3M="
1956 | },
1957 | "xmldom": {
1958 | "version": "0.1.27",
1959 | "resolved": "https://registry.npmjs.org/xmldom/-/xmldom-0.1.27.tgz",
1960 | "integrity": "sha1-1QH5ezvbQDr4757MIFcxh6rawOk="
1961 | },
1962 | "y18n": {
1963 | "version": "3.2.1",
1964 | "resolved": "https://registry.npmjs.org/y18n/-/y18n-3.2.1.tgz",
1965 | "integrity": "sha1-bRX7qITAhnnA136I53WegR4H+kE="
1966 | },
1967 | "yallist": {
1968 | "version": "2.1.2",
1969 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-2.1.2.tgz",
1970 | "integrity": "sha1-HBH5IY8HYImkfdUS+TxmmaaoHVI="
1971 | },
1972 | "yargs": {
1973 | "version": "11.0.0",
1974 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-11.0.0.tgz",
1975 | "integrity": "sha512-Rjp+lMYQOWtgqojx1dEWorjCofi1YN7AoFvYV7b1gx/7dAAeuI4kN5SZiEvr0ZmsZTOpDRcCqrpI10L31tFkBw==",
1976 | "requires": {
1977 | "cliui": "^4.0.0",
1978 | "decamelize": "^1.1.1",
1979 | "find-up": "^2.1.0",
1980 | "get-caller-file": "^1.0.1",
1981 | "os-locale": "^2.0.0",
1982 | "require-directory": "^2.1.1",
1983 | "require-main-filename": "^1.0.1",
1984 | "set-blocking": "^2.0.0",
1985 | "string-width": "^2.0.0",
1986 | "which-module": "^2.0.0",
1987 | "y18n": "^3.2.1",
1988 | "yargs-parser": "^9.0.2"
1989 | }
1990 | },
1991 | "yargs-parser": {
1992 | "version": "9.0.2",
1993 | "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-9.0.2.tgz",
1994 | "integrity": "sha1-nM9qQ0YP5O1Aqbto9I1DuKaMwHc=",
1995 | "requires": {
1996 | "camelcase": "^4.1.0"
1997 | }
1998 | },
1999 | "yauzl": {
2000 | "version": "2.4.1",
2001 | "resolved": "https://registry.npmjs.org/yauzl/-/yauzl-2.4.1.tgz",
2002 | "integrity": "sha1-lSj0QtqxsihOWLQ3m7GU4i4MQAU=",
2003 | "requires": {
2004 | "fd-slicer": "~1.0.1"
2005 | }
2006 | }
2007 | }
2008 | }
2009 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "react-styleguidist-visual",
3 | "description": "Automated visual testing for React Styleguidist, using Puppeteer and pixelmatch.",
4 | "version": "0.9.0",
5 | "license": "MIT",
6 | "author": {
7 | "name": "Daniel Perez Alvarez",
8 | "email": "unindented@gmail.com",
9 | "url": "http://unindented.org/"
10 | },
11 | "repository": {
12 | "type": "git",
13 | "url": "git://github.com/unindented/react-styleguidist-visual.git"
14 | },
15 | "main": "src/index.js",
16 | "bin": {
17 | "styleguidist-visual": "src/cli.js"
18 | },
19 | "scripts": {
20 | "test": "eslint ."
21 | },
22 | "dependencies": {
23 | "chalk": "^2.4.1",
24 | "debug": "^3.1.0",
25 | "fs-extra": "^6.0.1",
26 | "glob": "^7.1.2",
27 | "joi": "^13.4.0",
28 | "ora": "^2.1.0",
29 | "pixelmatch": "^4.0.2",
30 | "pngjs": "^3.3.3",
31 | "puppeteer": "^1.5.0",
32 | "term-img": "^2.1.0",
33 | "yargs": "^11.0.0"
34 | },
35 | "devDependencies": {
36 | "eslint": "4.19.1",
37 | "eslint-config-standard": "^11.0.0",
38 | "eslint-plugin-import": "^2.12.0",
39 | "eslint-plugin-node": "^6.0.1",
40 | "eslint-plugin-promise": "^3.8.0",
41 | "eslint-plugin-standard": "^3.1.0"
42 | }
43 | }
44 |
--------------------------------------------------------------------------------
/src/cli.js:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env node
2 | const yargs = require('yargs')
3 | const { test, approve } = require('.')
4 |
5 | const configParser = configPath => require(configPath)
6 |
7 | // eslint-disable-next-line no-unused-expressions
8 | yargs
9 | .command('test', 'Take screenshots of all examples in the specified URL.', testArgs, testCommand)
10 | .command('approve', 'Approve all new screenshots.', approveArgs, approveCommand)
11 | .demandCommand()
12 | .help()
13 | .wrap(null)
14 | .argv
15 |
16 | function testArgs (yargs) {
17 | yargs
18 | .option('url', {
19 | describe: 'URL to screenshot',
20 | type: 'string',
21 | requiresArg: true
22 | })
23 | .option('dir', {
24 | type: 'string',
25 | description: 'Directory where screenshots are stored'
26 | })
27 | .option('filter', {
28 | type: 'array',
29 | description: 'Only collect screenshots for these components'
30 | })
31 | .option('threshold', {
32 | type: 'number',
33 | description: 'Threshold for visual diffing'
34 | })
35 | .option('wait', {
36 | type: 'number',
37 | description: 'Delay before taking screenshot (in milliseconds)'
38 | })
39 | .demandOption(['url'])
40 | .config('config', configParser)
41 | .example(
42 | '$0 test --url "https://react-styleguidist.js.org/examples/basic/"',
43 | '# Take screenshots of all examples in the page'
44 | )
45 | .example(
46 | '$0 test --url "https://react-styleguidist.js.org/examples/basic/" --filter "Button"',
47 | '# Take screenshots of all examples for the "Button" component'
48 | )
49 | }
50 |
51 | function approveArgs (yargs) {
52 | yargs
53 | .option('dir', {
54 | type: 'string',
55 | description: 'Directory where screenshots are stored'
56 | })
57 | .option('filter', {
58 | type: 'array',
59 | description: 'Only approve screenshots for these components'
60 | })
61 | .config('config', configParser)
62 | .example('$0 approve', '# Approve all new screenshots')
63 | .example('$0 approve --filter "Button"', '# Approve all new screenshots for the "Button" component')
64 | }
65 |
66 | function testCommand (argv) {
67 | test(argv).catch(err => {
68 | console.log(err.message)
69 | process.exit(1)
70 | })
71 | }
72 |
73 | function approveCommand (argv) {
74 | approve(argv).catch(err => {
75 | console.log(err.message)
76 | process.exit(1)
77 | })
78 | }
79 |
--------------------------------------------------------------------------------
/src/commands/approve.js:
--------------------------------------------------------------------------------
1 | const joi = require('joi')
2 | const { getOptions } = require('../utils/options')
3 | const { promoteNewScreenshots } = require('../utils/image')
4 | const { debug } = require('../utils/debug')
5 |
6 | const approveSchema = joi
7 | .object()
8 | .unknown()
9 | .keys({
10 | dir: joi.string(),
11 | filter: joi.array().items(joi.string())
12 | })
13 |
14 | const approveDefaults = {
15 | dir: 'styleguide-visual',
16 | filter: undefined
17 | }
18 |
19 | async function approve (partialOptions) {
20 | try {
21 | const options = await getOptions(partialOptions, approveDefaults, approveSchema)
22 | const { dir, filter } = options
23 |
24 | await promoteNewScreenshots({ dir, filter })
25 | } catch (err) {
26 | debug(err)
27 | throw err
28 | }
29 | }
30 |
31 | module.exports = approve
32 |
--------------------------------------------------------------------------------
/src/commands/test.js:
--------------------------------------------------------------------------------
1 | const joi = require('joi')
2 | const puppeteer = require('puppeteer')
3 | const { getOptions } = require('../utils/options')
4 | const {
5 | checkForStaleRefScreenshots,
6 | compareNewScreenshotsToRefScreenshots,
7 | removeNonRefScreenshots
8 | } = require('../utils/image')
9 | const { getPreviews, takeNewScreenshotsOfPreviews } = require('../utils/page')
10 | const { debug, spinner } = require('../utils/debug')
11 |
12 | const testSchema = joi
13 | .object()
14 | .unknown()
15 | .keys({
16 | url: joi.string().required(),
17 | dir: joi.string(),
18 | filter: joi.array().items(joi.string()),
19 | threshold: joi
20 | .number()
21 | .min(0)
22 | .max(1),
23 | wait: joi.number(),
24 | viewports: joi.object().pattern(
25 | /^.+$/,
26 | joi.object().keys({
27 | width: joi
28 | .number()
29 | .integer()
30 | .min(1),
31 | height: joi
32 | .number()
33 | .integer()
34 | .min(1),
35 | deviceScaleFactor: joi
36 | .number()
37 | .integer()
38 | .min(1),
39 | isMobile: joi.boolean(),
40 | hasTouch: joi.boolean(),
41 | isLandscape: joi.boolean()
42 | })
43 | ),
44 | launchOptions: joi.object(),
45 | connectOptions: joi.object(),
46 | navigationOptions: joi.object()
47 | })
48 |
49 | const testDefaults = {
50 | url: undefined,
51 | sandbox: true,
52 | dir: 'styleguide-visual',
53 | filter: undefined,
54 | threshold: 0.001,
55 | deleteScreenshotsWhenAccepted: false,
56 | wait: 0,
57 | viewports: {
58 | desktop: {
59 | width: 800,
60 | height: 600,
61 | deviceScaleFactor: 1
62 | }
63 | },
64 | launchOptions: {},
65 | connectOptions: undefined,
66 | navigationOptions: {}
67 | }
68 |
69 | async function test (partialOptions) {
70 | let browser
71 | const useConnect = partialOptions.connectOptions !== undefined
72 |
73 | try {
74 | const options = await getOptions(partialOptions, testDefaults, testSchema)
75 | const { url, dir, filter, threshold, wait, viewports, launchOptions, connectOptions, navigationOptions, deleteScreenshotsWhenAccepted } = options
76 |
77 | await removeNonRefScreenshots({ dir, filter })
78 |
79 | browser = useConnect ? await puppeteer.connect(connectOptions) : await puppeteer.launch(launchOptions)
80 | const page = await browser.newPage()
81 |
82 | for (const viewport of Object.keys(viewports)) {
83 | const progress = spinner({
84 | start: `Taking screenshots for viewport ${viewport}`,
85 | update: `Taking screenshot of component %s of %s for viewport ${viewport}`,
86 | stop: `Finished taking screenshots for viewport ${viewport}`
87 | })
88 | progress.start()
89 | await page.setViewport(viewports[viewport])
90 | const previews = await getPreviews(page, { url, filter, viewport, navigationOptions })
91 | await takeNewScreenshotsOfPreviews(page, previews, { dir, progress, navigationOptions, wait })
92 | progress.stop()
93 | }
94 |
95 | await checkForStaleRefScreenshots({ dir, filter })
96 | await compareNewScreenshotsToRefScreenshots({ dir, filter, threshold, deleteScreenshotsWhenAccepted })
97 | } catch (err) {
98 | debug(err)
99 | throw err
100 | } finally {
101 | if (useConnect === false && browser != null) {
102 | await browser.close()
103 | }
104 | }
105 | }
106 |
107 | module.exports = test
108 |
--------------------------------------------------------------------------------
/src/index.js:
--------------------------------------------------------------------------------
1 | const test = require('./commands/test')
2 | const approve = require('./commands/approve')
3 |
4 | module.exports = {
5 | test,
6 | approve
7 | }
8 |
--------------------------------------------------------------------------------
/src/utils/debug.js:
--------------------------------------------------------------------------------
1 | const debug = require('debug')('react-styleguidist-visual')
2 | const chalk = require('chalk')
3 | const ora = require('ora')
4 | const { format } = require('util')
5 |
6 | const spinner = ({ start, update, stop }) => {
7 | const instance = ora({ color: 'white', text: start, enabled: debug.enabled ? false : undefined })
8 |
9 | return {
10 | start: () => {
11 | instance.text = start
12 | instance.start()
13 | },
14 | update: (curr, total) => {
15 | instance.text = format(update, curr, total)
16 | },
17 | stop: () => {
18 | instance.text = stop
19 | instance.stop()
20 | }
21 | }
22 | }
23 |
24 | const info = (text, ...args) => {
25 | console.log(` ${text}`, ...args)
26 | }
27 |
28 | const success = (text, ...args) => {
29 | console.log(format(`%s ${text}`, chalk.green('✔'), ...args))
30 | }
31 |
32 | const failure = (text, ...args) => {
33 | console.log(format(`%s ${text}`, chalk.red('✖'), ...args))
34 | }
35 |
36 | module.exports = {
37 | debug,
38 | spinner,
39 | info,
40 | success,
41 | failure
42 | }
43 |
--------------------------------------------------------------------------------
/src/utils/image.js:
--------------------------------------------------------------------------------
1 | const { promisify } = require('util')
2 | const fs = require('fs-extra')
3 | const path = require('path')
4 | const chalk = require('chalk')
5 | const globby = require('glob')
6 | const pixelmatch = require('pixelmatch')
7 | const { PNG } = require('pngjs')
8 | const termImg = require('term-img')
9 | const { debug, info, success, failure } = require('./debug')
10 |
11 | const move = promisify(fs.move)
12 | const pathExists = promisify(fs.pathExists)
13 | const remove = promisify(fs.remove)
14 | const glob = promisify(globby)
15 |
16 | async function checkForStaleRefScreenshots ({ dir, filter }) {
17 | const refImgs = await glob(path.join(dir, `${filter || ''}*.png`), {
18 | ignore: path.join(dir, `${filter || ''}*.{diff,new}.png`)
19 | })
20 |
21 | let hasStaleRefImgs = false
22 |
23 | for (const refImg of refImgs) {
24 | const newImg = refImg.replace(/\.png$/, '.new.png')
25 |
26 | const newImgExists = await pathExists(newImg)
27 | if (!newImgExists) {
28 | hasStaleRefImgs |= true
29 | failure('Screenshot %s is stale!', chalk.cyan(refImg))
30 | }
31 | }
32 |
33 | if (hasStaleRefImgs) {
34 | throw new Error('One or more screenshots are stale!')
35 | }
36 | }
37 |
38 | async function compareNewScreenshotsToRefScreenshots ({ dir, filter, threshold, deleteScreenshotsWhenAccepted }) {
39 | const newImgs = await glob(path.join(dir, `${filter || ''}*.new.png`))
40 |
41 | let diffCount = 0
42 |
43 | for (const newImg of newImgs) {
44 | const refImg = newImg.replace(/\.new\.png$/, '.png')
45 | const diffImg = newImg.replace(/\.new\.png$/, '.diff.png')
46 |
47 | const refImgExists = await pathExists(refImg)
48 | if (!refImgExists) {
49 | await promoteNewScreenshot(newImg)
50 | } else {
51 | const pixels = await diffScreenshots(newImg, refImg, diffImg, threshold)
52 | if (pixels === 0) {
53 | success('Screenshots %s and %s match', chalk.cyan(newImg), chalk.cyan(refImg))
54 | if (deleteScreenshotsWhenAccepted) {
55 | await remove(newImg)
56 | await remove(diffImg)
57 | }
58 | } else {
59 | failure('Screenshots %s and %s differ in %s pixels', chalk.cyan(newImg), chalk.cyan(refImg), chalk.red(pixels))
60 | termImg(diffImg, {
61 | fallback: () => {
62 | info('Check out the diff at %s', chalk.cyan(diffImg))
63 | }
64 | })
65 | diffCount++
66 | }
67 | }
68 | }
69 |
70 | if (diffCount > 0) {
71 | throw new Error('One or more new screenshots differ from their references!')
72 | }
73 | }
74 |
75 | async function promoteNewScreenshots ({ dir, filter }) {
76 | const newImgs = await glob(path.join(dir, `${filter || ''}*.new.png`))
77 | const oldDiffs = await glob(path.join(dir, `${filter || ''}*.diff.png`))
78 |
79 | for (const newImg of newImgs) {
80 | await promoteNewScreenshot(newImg)
81 | }
82 |
83 | for (const oldDiff of oldDiffs) {
84 | await remove(oldDiff)
85 | }
86 | }
87 |
88 | async function promoteNewScreenshot (newImg) {
89 | const refImg = newImg.replace(/\.new\.png$/, '.png')
90 |
91 | debug('Promoting screenshot from %s to %s', chalk.cyan(newImg), chalk.cyan(refImg))
92 |
93 | return move(newImg, refImg, { overwrite: true })
94 | }
95 |
96 | async function diffScreenshots (img1, img2, output, threshold = 0.001) {
97 | debug('Diffing screenshots %s and %s', chalk.cyan(img1), chalk.cyan(img2))
98 |
99 | const [data1, data2] = await Promise.all([readScreenshot(img1), readScreenshot(img2)])
100 | const diff = new PNG({ width: data1.width, height: data1.height })
101 | const pixels =
102 | pixelmatch(data1.data, data2.data, diff.data, data1.width, data1.height, {
103 | threshold
104 | }) || Math.abs(data1.data.length - data2.data.length)
105 |
106 | debug('Screenshots %s and %s differ in %s pixels', chalk.cyan(img1), chalk.cyan(img2), chalk.red(pixels))
107 |
108 | return new Promise((resolve, reject) => {
109 | diff
110 | .pack()
111 | .pipe(fs.createWriteStream(output))
112 | .on('finish', function () {
113 | resolve(pixels)
114 | })
115 | .on('error', reject)
116 | })
117 | }
118 |
119 | async function readScreenshot (img) {
120 | return new Promise((resolve, reject) => {
121 | fs
122 | .createReadStream(img)
123 | .pipe(new PNG())
124 | .on('parsed', function () {
125 | resolve(this)
126 | })
127 | .on('error', reject)
128 | })
129 | }
130 |
131 | async function removeNonRefScreenshots ({ dir, filter }) {
132 | const nonRefImgs = await glob(path.join(dir, `${filter || ''}*.{diff,new}.png`))
133 |
134 | for (const nonRefImg of nonRefImgs) {
135 | await remove(nonRefImg)
136 | }
137 | }
138 |
139 | module.exports = {
140 | checkForStaleRefScreenshots,
141 | compareNewScreenshotsToRefScreenshots,
142 | promoteNewScreenshots,
143 | removeNonRefScreenshots
144 | }
145 |
--------------------------------------------------------------------------------
/src/utils/options.js:
--------------------------------------------------------------------------------
1 | const { promisify } = require('util')
2 | const joi = require('joi')
3 |
4 | const validate = promisify(joi.validate)
5 |
6 | async function getOptions (options, defaults, schema) {
7 | const optionsWithDefaults = Object.assign({}, cleanOptions(defaults), cleanOptions(options))
8 | return validate(optionsWithDefaults, schema)
9 | }
10 |
11 | function cleanOptions (options) {
12 | return Object.keys(options).reduce((memo, key) => {
13 | if (options[key] != null) {
14 | memo[key] = options[key]
15 | }
16 | return memo
17 | }, {})
18 | }
19 |
20 | module.exports = {
21 | getOptions
22 | }
23 |
--------------------------------------------------------------------------------
/src/utils/page.js:
--------------------------------------------------------------------------------
1 | const { promisify } = require('util')
2 | const fs = require('fs-extra')
3 | const path = require('path')
4 | const chalk = require('chalk')
5 | const { debug } = require('./debug')
6 |
7 | const ensureDir = promisify(fs.ensureDir)
8 |
9 | async function getPreviews (page, { url, filter, viewport, navigationOptions }) {
10 | await goToUrl(page, url, navigationOptions)
11 |
12 | return page.evaluate(getPreviewsInPage, { filter, viewport })
13 | }
14 |
15 | function getPreviewsInPage ({ filter, viewport }) {
16 | const shouldIncludePreview = name => {
17 | if (filter == null) {
18 | return true
19 | }
20 |
21 | return [].concat(filter).some(str => {
22 | const regexp = new RegExp(str.toLowerCase())
23 | return regexp.test(name.toLowerCase())
24 | })
25 | }
26 |
27 | const extractPreviewInfo = (memo, el) => {
28 | const url = el.nextSibling.querySelector('a[href][title]').href
29 | const name = el.dataset.preview
30 | const description = el.dataset.description
31 | const actionStates = el.dataset.actionStates
32 | const previewSelector = el.dataset.previewSelector
33 |
34 | if (!shouldIncludePreview(name)) {
35 | return memo
36 | }
37 |
38 | memo[name] = (memo[name] || []).concat({
39 | url,
40 | name,
41 | description,
42 | actionStates,
43 | viewport,
44 | previewSelector
45 | })
46 |
47 | return memo
48 | }
49 |
50 | const result = document.querySelectorAll('[data-preview]')
51 | return Array.prototype.reduce.call(result, extractPreviewInfo, {})
52 | }
53 |
54 | async function takeNewScreenshotsOfPreviews (page, previewMap, { dir, progress, navigationOptions, wait }) {
55 | await ensureDir(dir)
56 |
57 | let progressIndex = 1
58 | const progressTotal = Object.keys(previewMap).reduce((memo, name) => memo + previewMap[name].length, 0)
59 |
60 | for (const name of Object.keys(previewMap)) {
61 | const previewList = previewMap[name]
62 |
63 | let previewIndex = 1
64 |
65 | for (const preview of previewList) {
66 | const actionStateList = preview.actionStates ? JSON.parse(preview.actionStates) : [{ action: 'none' }]
67 |
68 | progress.update(progressIndex, progressTotal)
69 |
70 | const { url } = preview
71 | await goToHashUrl(page, url)
72 |
73 | for (const actionState of actionStateList) {
74 | await takeNewScreenshotOfPreview(page, preview, previewIndex, actionState, { dir, wait })
75 | previewIndex += 1
76 | }
77 | await resetMouseAndFocus(page)
78 | progressIndex += 1
79 | }
80 | }
81 | }
82 |
83 | async function takeNewScreenshotOfPreview (page, preview, index, actionState, { dir, wait }) {
84 | const el = await page.$('[data-preview]')
85 |
86 | if (wait) {
87 | await sleep(wait)
88 | }
89 |
90 | await triggerAction(page, el, actionState)
91 |
92 | const boundingBoxEl = preview.previewSelector ? await page.$(preview.previewSelector) : el
93 | const boundingBox = await boundingBoxEl.boundingBox()
94 |
95 | const path = await getRelativeFilepath(preview, index, actionState, dir)
96 | debug('Storing screenshot of %s in %s', chalk.blue(preview.name), chalk.cyan(path))
97 | await page.screenshot({ clip: boundingBox, path })
98 | }
99 |
100 | async function getRelativeFilepath (preview, index, actionState, dir) {
101 | const { name, description = `${index}`, viewport } = preview
102 | const { action = '', key = '' } = actionState
103 | const actionName = action === 'none' ? '' : action
104 | const baseName = name + ` ${description} ${actionName} ${key} ${viewport}`.toLowerCase()
105 | const underscoredName = baseName.replace(/[^0-9A-Z]+/gi, '_')
106 |
107 | return path.join(dir, `${underscoredName}.new.png`)
108 | }
109 |
110 | async function triggerAction (page, el, actionState) {
111 | const actionEl = actionState.selector ? await el.$(actionState.selector) || await page.$(actionState.selector) : el
112 | switch (actionState.action) {
113 | case 'hover':
114 | await actionEl.hover()
115 | break
116 | case 'click':
117 | await actionEl.click()
118 | break
119 | case 'mouseDown':
120 | const box = await actionEl.boundingBox()
121 | await page.mouse.move(box.x + box.width / 2, box.y + box.height / 2)
122 | await page.mouse.down()
123 | break
124 | case 'focus':
125 | await actionEl.focus()
126 | break
127 | case 'keyPress':
128 | const key = actionState.key || 'a'
129 | await page.keyboard.press(key)
130 | break
131 | }
132 | await sleep(actionState.wait)
133 | }
134 |
135 | async function resetMouseAndFocus (page) {
136 | await page.focus('body')
137 | await page.mouse.up()
138 | await page.mouse.move(0, 0)
139 | }
140 |
141 | async function goToUrl (page, url, navigationOptions) {
142 | debug('Navigating to URL %s', chalk.blue(url))
143 | return page.goto(url, navigationOptions)
144 | }
145 |
146 | async function goToHashUrl (page, url) {
147 | debug('Navigating to hash URL %s', chalk.blue(url))
148 | return page.evaluate(url => {
149 | window.location.href = url
150 | }, url)
151 | }
152 |
153 | function sleep (ms) {
154 | return new Promise(resolve => setTimeout(resolve, ms))
155 | }
156 |
157 | module.exports = {
158 | getPreviews,
159 | takeNewScreenshotsOfPreviews
160 | }
161 |
--------------------------------------------------------------------------------