diff --git a/.drone.yml b/.drone.yml index 1cee7dc2d..4d06e044a 100644 --- a/.drone.yml +++ b/.drone.yml @@ -24,6 +24,16 @@ pipeline: commands: - go test -timeout 30s github.com/laszlocph/woodpecker/store/datastore + test_frontend: + image: node:10.17.0-stretch + commands: + - make test-frontend + + build_frontend: + image: node:10.17.0-stretch + commands: + - make build-frontend + build: image: golang:1.12.4 commands: sh .drone.sh diff --git a/.gitignore b/.gitignore index e2cebad11..4aa06b940 100644 --- a/.gitignore +++ b/.gitignore @@ -10,3 +10,8 @@ cli/release/ server/swagger/files/*.json server/swagger/swagger_gen.go .idea/ + +web/node_modules +web/dist/files +web/*.log +web/.env diff --git a/Makefile b/Makefile index 5165f581c..cbfa938b2 100644 --- a/Makefile +++ b/Makefile @@ -26,6 +26,9 @@ test-agent: test-server: $(DOCKER_RUN) go test -race -timeout 30s github.com/laszlocph/woodpecker/cmd/drone-server +test-frontend: + (cd web/; yarn run test) + test-lib: $(DOCKER_RUN) go test -race -timeout 30s $(shell go list ./... | grep -v '/cmd/') @@ -37,6 +40,10 @@ build-agent: build-server: $(DOCKER_RUN) go build -o build/drone-server github.com/laszlocph/woodpecker/cmd/drone-server +build-frontend: + (cd web/; yarn run build) + + build: build-agent build-server install: diff --git a/go.mod b/go.mod index 1c68b6718..00e4be179 100644 --- a/go.mod +++ b/go.mod @@ -35,7 +35,6 @@ require ( github.com/joho/godotenv v0.0.0-20150907010228-4ed13390c0ac github.com/kr/pretty v0.0.0-20160708215748-737b74a46c4b github.com/kr/text v0.0.0-20160504234017-7cafcd837844 // indirect - github.com/laszlocph/woodpecker-ui v0.0.0-20190724123035-653e49c05045 github.com/lib/pq v0.0.0-20151015211310-83c4f410d0ae github.com/manucorporat/sse v0.0.0-20160126180136-ee05b128a739 // indirect github.com/mattn/go-sqlite3 v0.0.0-20170901084005-05548ff55570 diff --git a/go.sum b/go.sum index af8b31bb7..d3fcb7606 100644 --- a/go.sum +++ b/go.sum @@ -79,8 +79,6 @@ github.com/kr/pretty v0.0.0-20160708215748-737b74a46c4b h1:LJ9zj3Zit+pLjAQtA1gxl github.com/kr/pretty v0.0.0-20160708215748-737b74a46c4b/go.mod h1:Bvhd+E3laJ0AVkG0c9rmtZcnhV0HQ3+c3YxxqTvc/gA= github.com/kr/text v0.0.0-20160504234017-7cafcd837844 h1:kpzneEBeC0dMewP3gr/fADv1OlblH9r1goWVwpOt3TU= github.com/kr/text v0.0.0-20160504234017-7cafcd837844/go.mod h1:sjUstKUATFIcff4qlB53Kml0wQPtJVc/3fWrmuUmcfA= -github.com/laszlocph/woodpecker-ui v0.0.0-20190724123035-653e49c05045 h1:H2vySVhUS29MOkY1tdn9NOPsffndAf2/5kod0gwX1m4= -github.com/laszlocph/woodpecker-ui v0.0.0-20190724123035-653e49c05045/go.mod h1:+Ly1/ou5jW3YIvsEiJWNJTIe3Zt8dLPd0cgrmbjARoE= github.com/lib/pq v0.0.0-20151015211310-83c4f410d0ae h1:rBqRT7VqVLePKGtyV6xDFLXeqD56CvZKEqI0XWzVTxM= github.com/lib/pq v0.0.0-20151015211310-83c4f410d0ae/go.mod h1:5WUZQaWbwv1U+lTReE5YruASi9Al49XbQIvNi/34Woo= github.com/manucorporat/sse v0.0.0-20160126180136-ee05b128a739 h1:ykXz+pRRTibcSjG1yRhpdSHInF8yZY/mfn+Rz2Nd1rE= diff --git a/server/web/web.go b/server/web/web.go index 43e322b66..9cc6aef51 100644 --- a/server/web/web.go +++ b/server/web/web.go @@ -24,7 +24,7 @@ import ( "path/filepath" "time" - "github.com/laszlocph/woodpecker-ui/dist" + "github.com/laszlocph/woodpecker/web/dist" "github.com/laszlocph/woodpecker/model" "github.com/laszlocph/woodpecker/shared/token" "github.com/laszlocph/woodpecker/version" diff --git a/web/.babelrc b/web/.babelrc new file mode 100644 index 000000000..fd95ce5a9 --- /dev/null +++ b/web/.babelrc @@ -0,0 +1,16 @@ +{ + "sourceMaps": false, + "presets": [ + ["es2015", { "loose":true }], + "stage-0", + "react" + ], + "plugins": [ + ["transform-decorators-legacy"], + ["transform-object-rest-spread"], + ["transform-react-jsx"], + ["transform-es3-property-literals"], + ["transform-es3-member-expression-literals"], + ["transform-decorators-legacy"] + ] +} diff --git a/web/.drone.yml b/web/.drone.yml new file mode 100644 index 000000000..cabcbe8bf --- /dev/null +++ b/web/.drone.yml @@ -0,0 +1,9 @@ +pipeline: + build: + image: node:8 + commands: + - yarn install + - yarn run lesshint + - yarn run test + - yarn run lint --quiet + - yarn run build diff --git a/web/.eslintrc.js b/web/.eslintrc.js new file mode 100644 index 000000000..c778afbdf --- /dev/null +++ b/web/.eslintrc.js @@ -0,0 +1,38 @@ +module.exports = { + "extends": [ + "standard", + "plugin:jest/recommended", + "plugin:react/recommended", + "prettier", + "prettier/react" + ], + "plugins": [ + "react", + "jest", + "prettier" + ], + "parser": "babel-eslint", + "parserOptions": { + "ecmaVersion": 2016, + "sourceType": "module", + "ecmaFeatures": { + "jsx": true + } + }, + "env": { + "es6": true, + "browser": true, + "node": true, + "jest/globals": true + }, + "rules": { + "react/prop-types": 1, + "prettier/prettier": [ + "error", + { + "trailingComma": "all", + "useTabs": true + } + ] + } +}; diff --git a/web/.lesshintrc b/web/.lesshintrc new file mode 100644 index 000000000..9ad09a8ef --- /dev/null +++ b/web/.lesshintrc @@ -0,0 +1,17 @@ +{ + "fileExtensions": [".less", ".css"], + + "excludedFiles": ["ansi.less"], + + "spaceAfterPropertyColon": { + "enabled": true, + "style": "one_space" + }, + + "emptyRule": true, + "qualifyingElement": false, + "trailingWhitespace": true, + "zeroUnit": { + "exclude": ["flex"] + } +} diff --git a/web/LICENSE b/web/LICENSE new file mode 100644 index 000000000..972fb5cc0 --- /dev/null +++ b/web/LICENSE @@ -0,0 +1,20 @@ +Copyright 2017 Drone.IO Inc +Copyright 2019 Laszlo Fogas + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. + +--- + +Woodpecker icon by Georgiana Ionescu from the Noun Project +Licensed as Creative Commons CCBY +https://thenounproject.com/term/woodpecker/1761314/ diff --git a/web/README.md b/web/README.md new file mode 100644 index 000000000..64b4d1a0f --- /dev/null +++ b/web/README.md @@ -0,0 +1,57 @@ +This project contains the source code for the drone user interface. The generated javascript and css assets are embedded into a Go source file which is imported into the main drone application, using go get. + +## Building + +To compile the source and create minified css and javascript assets: + +```text +yarn install # install project dependencies + +yarn run format # formats the codebase +yarn run lint # lints the codebase +yarn run test # tests the codebase +yarn run build # builds the production bundle +``` + +## Running + +To run a devserver with watching, hotreloading and proxy to drone server: + +```text +export DRONE_SERVER= +export DRONE_TOKEN= + +yarn run start +``` + +For example: + +```text +export DRONE_SERVER=http://your.drone.server +export DRONE_TOKEN=eyJhbGciOiJIUzI1NiIsIn... + +yarn run start +``` + +Note you will need to retrieve your drone user token from the tokens screen in the drone user interface. When the server is running you can open the following url in your browser: + +```text +http://localhost:9999 +``` + +## Releases + +To bundle and embed the code in a Go source file install the following command line utility: + +```text +go get github.com/bradrydzewski/togo +``` + +To generate the Go source file run the following command: + +```text +go generate ./... +go install ./... +``` + +__Note__ that for security reasons we will not accept a pull request that updates embedded Go asset file since we are not able to easily review the embedded, minified code. This file is instead automatically generated by our build server to prevent tampering. diff --git a/web/dist/dist.go b/web/dist/dist.go new file mode 100644 index 000000000..7fb13feee --- /dev/null +++ b/web/dist/dist.go @@ -0,0 +1,3 @@ +package dist + +//go:generate togo http -package dist -output dist_gen.go diff --git a/web/dist/dist_gen.go b/web/dist/dist_gen.go new file mode 100644 index 000000000..0c6da8e36 --- /dev/null +++ b/web/dist/dist_gen.go @@ -0,0 +1,11042 @@ +package dist + +import ( + "bytes" + "net/http" + "os" + "strings" + "time" +) + +type fileSystem struct { + files map[string]file +} + +func (fs *fileSystem) Open(name string) (http.File, error) { + name = strings.Replace(name, "//", "/", -1) + f, ok := fs.files[name] + if ok { + return newHTTPFile(f, false), nil + } + index := strings.Replace(name+"/index.html", "//", "/", -1) + f, ok = fs.files[index] + if !ok { + return nil, os.ErrNotExist + } + return newHTTPFile(f, true), nil +} + +type file struct { + os.FileInfo + data []byte +} + +type fileInfo struct { + name string + size int64 + mode os.FileMode + modTime time.Time + isDir bool + + files []os.FileInfo +} + +func (f *fileInfo) Name() string { + return f.name +} + +func (f *fileInfo) Size() int64 { + return f.size +} + +func (f *fileInfo) Mode() os.FileMode { + return f.mode +} + +func (f *fileInfo) ModTime() time.Time { + return f.modTime +} + +func (f *fileInfo) IsDir() bool { + return f.isDir +} + +func (f *fileInfo) Readdir(count int) ([]os.FileInfo, error) { + return make([]os.FileInfo, 0), nil +} + +func (f *fileInfo) Sys() interface{} { + return nil +} + +func newHTTPFile(file file, isDir bool) *httpFile { + return &httpFile{ + file: file, + reader: bytes.NewReader(file.data), + isDir: isDir, + } +} + +type httpFile struct { + file + + reader *bytes.Reader + isDir bool +} + +func (f *httpFile) Read(p []byte) (n int, err error) { + return f.reader.Read(p) +} + +func (f *httpFile) Seek(offset int64, whence int) (ret int64, err error) { + return f.reader.Seek(offset, whence) +} + +func (f *httpFile) Stat() (os.FileInfo, error) { + return f, nil +} + +func (f *httpFile) IsDir() bool { + return f.isDir +} + +func (f *httpFile) Readdir(count int) ([]os.FileInfo, error) { + return make([]os.FileInfo, 0), nil +} + +func (f *httpFile) Close() error { + return nil +} + +// New returns an embedded http.FileSystem +func New() http.FileSystem { + return &fileSystem{ + files: files, + } +} + +// Lookup returns the file at the specified path +func Lookup(path string) ([]byte, error) { + f, ok := files[path] + if !ok { + return nil, os.ErrNotExist + } + return f.data, nil +} + +// MustLookup returns the file at the specified path +// and panics if the file is not found. +func MustLookup(path string) []byte { + d, err := Lookup(path) + if err != nil { + panic(err) + } + return d +} + +// Index of all files +var files = map[string]file{ + "/static/vendor.1befdccc9ddcaf56a7d3.js": { + data: file0, + FileInfo: &fileInfo{ + name: "vendor.1befdccc9ddcaf56a7d3.js", + size: 272277, + modTime: time.Unix(1573564669, 0), + }, + }, + "/static/bundle.2fbf04cbe6d89cbe017f.js": { + data: file1, + FileInfo: &fileInfo{ + name: "bundle.2fbf04cbe6d89cbe017f.js", + size: 374464, + modTime: time.Unix(1573564669, 0), + }, + }, + "/favicon.png": { + data: file2, + FileInfo: &fileInfo{ + name: "favicon.png", + size: 1374, + modTime: time.Unix(1573564669, 0), + }, + }, + "/index.html": { + data: file3, + FileInfo: &fileInfo{ + name: "index.html", + size: 388, + modTime: time.Unix(1573564669, 0), + }, + }, +} + +// +// embedded files. +// + +// /static/vendor.1befdccc9ddcaf56a7d3.js +var file0 = []byte(`!function(t){function e(n){if(r[n])return r[n].exports;var o=r[n]={i:n,l:!1,exports:{}};return t[n].call(o.exports,o,o.exports,e),o.l=!0,o.exports}var n=window.webpackJsonp;window.webpackJsonp=function(r,i,a){for(var u,c,s,f=0,l=[];r.length>f;f++)c=r[f],o[c]&&l.push(o[c][0]),o[c]=0;for(u in i)Object.prototype.hasOwnProperty.call(i,u)&&(t[u]=i[u]);for(n&&n(r,i,a);l.length;)l.shift()();if(a)for(f=0;a.length>f;f++)s=e(e.s=a[f]);return s};var r={},o={1:0};e.e=function(t){function n(){u.onerror=u.onload=null,clearTimeout(c);var e=o[t];0!==e&&(e&&e[1](Error("Loading chunk "+t+" failed.")),o[t]=void 0)}var r=o[t];if(0===r)return new Promise(function(t){t()});if(r)return r[2];var i=new Promise(function(e,n){r=o[t]=[e,n]});r[2]=i;var a=document.getElementsByTagName("head")[0],u=document.createElement("script");u.type="text/javascript",u.charset="utf-8",u.async=!0,u.timeout=12e4,e.nc&&u.setAttribute("nonce",e.nc),u.src=e.p+""+t+".static/bundle."+{0:"2fbf04cbe6d89cbe017f"}[t]+".js";var c=setTimeout(n,12e4);return u.onerror=u.onload=n,a.appendChild(u),i},e.m=t,e.c=r,e.d=function(t,n,r){e.o(t,n)||Object.defineProperty(t,n,{configurable:!1,enumerable:!0,get:r})},e.n=function(t){var n=t&&t.__esModule?function(){return t.default}:function(){return t};return e.d(n,"a",n),n},e.o=function(t,e){return Object.prototype.hasOwnProperty.call(t,e)},e.p="/",e.oe=function(t){throw t},e(e.s=584)}([function(t,e,n){var r=n(5),o=n(32),i=n(18),a=n(19),u=n(28),c=function(t,e,n){var s,f,l,p,h=t&c.F,d=t&c.G,v=t&c.S,y=t&c.P,m=t&c.B,g=d?r:v?r[e]||(r[e]={}):(r[e]||{}).prototype,b=d?o:o[e]||(o[e]={}),_=b.prototype||(b.prototype={});d&&(n=e);for(s in n)f=!h&&g&&void 0!==g[s],l=(f?g:n)[s],p=m&&f?u(l,r):y&&"function"==typeof l?u(Function.call,l):l,g&&a(g,s,l,t&c.U),b[s]!=l&&i(b,s,p),y&&_[s]!=l&&(_[s]=l)};r.core=o,c.F=1,c.G=2,c.S=4,c.P=8,c.B=16,c.W=32,c.U=64,c.R=128,t.exports=c},function(t,e,n){(function(e){!function(e,r){t.exports=r(n(12),n(118))}(0,function(t,n){function r(){return null}function o(t){var e=t.nodeName,n=t.attributes;t.attributes={},e.defaultProps&&w(t.attributes,e.defaultProps),n&&w(t.attributes,n)}function i(t,e){var n,r,o;if(e){for(o in e)if(n=B.test(o))break;if(n){r=t.attributes={};for(o in e)e.hasOwnProperty(o)&&(r[B.test(o)?o.replace(/([A-Z0-9])/,"-$1").toLowerCase():o]=e[o])}}}function a(t,e,r){var o=e&&e._preactCompatRendered&&e._preactCompatRendered.base;o&&o.parentNode!==e&&(o=null),!o&&e&&(o=e.firstElementChild);for(var i=e.childNodes.length;i--;)e.childNodes[i]!==o&&e.removeChild(e.childNodes[i]);var a=n.render(t,e,o);return e&&(e._preactCompatRendered=a&&(a._component||{base:a})),"function"==typeof r&&r(),a&&a._component||a}function u(t,e,r,o){var i=n.h(Y,{context:t.context},e),u=a(i,r),c=u._component||u.base;return o&&o.call(c,u),c}function c(t){var e=t._preactCompatRendered&&t._preactCompatRendered.base;return!(!e||e.parentNode!==t)&&(n.render(n.h(r),t,e),!0)}function s(t){return d.bind(null,t)}function f(t,e){for(var n=e||0;t.length>n;n++){var r=t[n];Array.isArray(r)?f(r):r&&"object"==typeof r&&!m(r)&&(r.props&&r.type||r.attributes&&r.nodeName||r.children)&&(t[n]=d(r.type||r.nodeName,r.props||r.attributes,r.children))}}function l(t){return"function"==typeof t&&!(t.prototype&&t.prototype.render)}function p(t){return E({displayName:t.displayName||t.name,render:function(){return t(this.props,this.context)}})}function h(t){var e=t[U];return e?!0===e?t:e:(e=p(t),Object.defineProperty(e,U,{configurable:!0,value:!0}),e.displayName=t.displayName,e.propTypes=t.propTypes,e.defaultProps=t.defaultProps,Object.defineProperty(t,U,{configurable:!0,value:e}),e)}function d(){for(var t=[],e=arguments.length;e--;)t[e]=arguments[e];return f(t,2),v(n.h.apply(void 0,t))}function v(t){t.preactCompatNormalized=!0,_(t),l(t.nodeName)&&(t.nodeName=h(t.nodeName));var e=t.attributes.ref,n=e&&typeof e;return!$||"string"!==n&&"number"!==n||(t.attributes.ref=g(e,$)),b(t),t}function y(t,e){for(var r=[],o=arguments.length-2;o-- >0;)r[o]=arguments[o+2];if(!m(t))return t;var i=t.attributes||t.props,a=n.h(t.nodeName||t.type,i,t.children||i&&i.children),u=[a,e];return r&&r.length?u.push(r):e&&e.children&&u.push(e.children),v(n.cloneElement.apply(void 0,u))}function m(t){return t&&(t instanceof V||t.$$typeof===D)}function g(t,e){return e._refProxies[t]||(e._refProxies[t]=function(n){e&&e.refs&&(e.refs[t]=n,null===n&&(delete e._refProxies[t],e=null))})}function b(t){var e=t.nodeName,n=t.attributes;if(n&&"string"==typeof e){var r={};for(var o in n)r[o.toLowerCase()]=o;if(r.ondoubleclick&&(n.ondblclick=n[r.ondoubleclick],delete n[r.ondoubleclick]),r.onchange&&("textarea"===e||"input"===e.toLowerCase()&&!/^fil|che|rad/i.test(n.type))){var i=r.oninput||"oninput";n[i]||(n[i]=C([n[i],n[r.onchange]]),delete n[r.onchange])}}}function _(t){var e=t.attributes||(t.attributes={});Z.enumerable="className"in e,e.className&&(e.class=e.className),Object.defineProperty(e,"className",Z)}function w(t){for(var e=arguments,n=1,r=void 0;arguments.length>n;n++)if(r=e[n])for(var o in r)r.hasOwnProperty(o)&&(t[o]=r[o]);return t}function O(t,e){for(var n in t)if(!(n in e))return!0;for(var r in e)if(t[r]!==e[r])return!0;return!1}function x(t){return t&&t.base||t}function P(){}function E(t){function e(t,e){S(this),L.call(this,t,e,z),M.call(this,t,e)}return t=w({constructor:e},t),t.mixins&&j(t,k(t.mixins)),t.statics&&w(e,t.statics),t.propTypes&&(e.propTypes=t.propTypes),t.defaultProps&&(e.defaultProps=t.defaultProps),t.getDefaultProps&&(e.defaultProps=t.getDefaultProps()),P.prototype=L.prototype,e.prototype=w(new P,t),e.displayName=t.displayName||"Component",e}function k(t){for(var e={},n=0;t.length>n;n++){var r=t[n];for(var o in r)r.hasOwnProperty(o)&&"function"==typeof r[o]&&(e[o]||(e[o]=[])).push(r[o])}return e}function j(t,e){for(var n in e)e.hasOwnProperty(n)&&(t[n]=C(e[n].concat(t[n]||K),"getDefaultProps"===n||"getInitialState"===n||"getChildContext"===n))}function S(t){for(var e in t){var n=t[e];"function"!=typeof n||n.__bound||W.hasOwnProperty(e)||((t[e]=n.bind(t)).__bound=!0)}}function T(t,e,n){if("string"==typeof e&&(e=t.constructor.prototype[e]),"function"==typeof e)return e.apply(t,n)}function C(t,e){return function(){for(var n,r=arguments,o=this,i=0;t.length>i;i++){var a=T(o,t[i],r);if(e&&null!=a){n||(n={});for(var u in a)a.hasOwnProperty(u)&&(n[u]=a[u])}else void 0!==a&&(n=a)}return n}}function M(t,e){N.call(this,t,e),this.componentWillReceiveProps=C([N,this.componentWillReceiveProps||"componentWillReceiveProps"]),this.render=C([N,R,this.render||"render",A])}function N(e){if(e){var n=e.children;if(n&&Array.isArray(n)&&1===n.length&&("string"==typeof n[0]||"function"==typeof n[0]||n[0]instanceof V)&&(e.children=n[0])&&"object"==typeof e.children&&(e.children.length=1,e.children[0]=e.children),q){var r="function"==typeof this?this:this.constructor,o=this.propTypes||r.propTypes,i=this.displayName||r.name;o&&t.checkPropTypes(o,e,"prop",i)}}}function R(){$=this}function A(){$===this&&($=null)}function L(t,e,r){n.Component.call(this,t,e),this.state=this.getInitialState?this.getInitialState():{},this.refs={},this._refProxies={},r!==z&&M.call(this,t,e)}function I(t,e){L.call(this,t,e)}t="default"in t?t.default:t;var F="a abbr address area article aside audio b base bdi bdo big blockquote body br button canvas caption cite code col colgroup data datalist dd del details dfn dialog div dl dt em embed fieldset figcaption figure footer form h1 h2 h3 h4 h5 h6 head header hgroup hr html i iframe img input ins kbd keygen label legend li link main map mark menu menuitem meta meter nav noscript object ol optgroup option output p param picture pre progress q rp rt ruby s samp script section select small source span strong style sub summary sup table tbody td textarea tfoot th thead time title tr track u ul var video wbr circle clipPath defs ellipse g image line linearGradient mask path pattern polygon polyline radialGradient rect stop svg text tspan".split(" "),D="undefined"!=typeof Symbol&&Symbol.for&&Symbol.for("react.element")||60103,U="undefined"!=typeof Symbol?Symbol.for("__preactCompatWrapper"):"__preactCompatWrapper",W={constructor:1,render:1,shouldComponentUpdate:1,componentWillReceiveProps:1,componentWillUpdate:1,componentDidUpdate:1,componentWillMount:1,componentDidMount:1,componentWillUnmount:1,componentDidUnmount:1},B=/^(?:accent|alignment|arabic|baseline|cap|clip|color|fill|flood|font|glyph|horiz|marker|overline|paint|stop|strikethrough|stroke|text|underline|unicode|units|v|vector|vert|word|writing|x)[A-Z]/,z={},q=void 0===e||!e.env||"production"!==e.env.NODE_ENV,V=n.h("a",null).constructor;V.prototype.$$typeof=D,V.prototype.preactCompatUpgraded=!1,V.prototype.preactCompatNormalized=!1,Object.defineProperty(V.prototype,"type",{get:function(){return this.nodeName},set:function(t){this.nodeName=t},configurable:!0}),Object.defineProperty(V.prototype,"props",{get:function(){return this.attributes},set:function(t){this.attributes=t},configurable:!0});var G=n.options.event;n.options.event=function(t){return G&&(t=G(t)),t.persist=Object,t.nativeEvent=t,t};var H=n.options.vnode;n.options.vnode=function(t){if(!t.preactCompatUpgraded){t.preactCompatUpgraded=!0;var e=t.nodeName,n=t.attributes=w({},t.attributes);"function"==typeof e?(!0===e[U]||e.prototype&&"isReactComponent"in e.prototype)&&(t.children&&t.children+""==""&&(t.children=void 0),t.children&&(n.children=t.children),t.preactCompatNormalized||v(t),o(t)):(t.children&&t.children+""==""&&(t.children=void 0),t.children&&(n.children=t.children),n.defaultValue&&(n.value||0===n.value||(n.value=n.defaultValue),delete n.defaultValue),i(t,n))}H&&H(t)};var Y=function(){};Y.prototype.getChildContext=function(){return this.props.context},Y.prototype.render=function(t){return t.children[0]};for(var $,K=[],J={map:function(t,e,n){return null==t?null:(t=J.toArray(t),n&&n!==t&&(e=e.bind(n)),t.map(e))},forEach:function(t,e,n){if(null==t)return null;t=J.toArray(t),n&&n!==t&&(e=e.bind(n)),t.forEach(e)},count:function(t){return t&&t.length||0},only:function(t){if(t=J.toArray(t),1!==t.length)throw Error("Children.only() expects only one child.");return t[0]},toArray:function(t){return null==t?[]:K.concat(t)}},Q={},X=F.length;X--;)Q[F[X]]=s(F[X]);var Z={configurable:!0,get:function(){return this.class},set:function(t){this.class=t}};return w(L.prototype=new n.Component,{constructor:L,isReactComponent:{},replaceState:function(t,e){var n=this;this.setState(t,e);for(var r in n.state)r in t||delete n.state[r]},getDOMNode:function(){return this.base},isMounted:function(){return!!this.base}}),P.prototype=L.prototype,I.prototype=new P,I.prototype.isPureReactComponent=!0,I.prototype.shouldComponentUpdate=function(t,e){return O(this.props,t)||O(this.state,e)},{version:"15.1.0",DOM:Q,PropTypes:t,Children:J,render:a,createClass:E,createFactory:s,createElement:d,cloneElement:y,isValidElement:m,findDOMNode:x,unmountComponentAtNode:c,Component:L,PureComponent:I,unstable_renderSubtreeIntoContainer:u,__spread:w}})}).call(e,n(21))},function(t,e,n){var r=n(7);t.exports=function(t){if(!r(t))throw TypeError(t+" is not an object!");return t}},,,function(t){var e=t.exports="undefined"!=typeof window&&window.Math==Math?window:"undefined"!=typeof self&&self.Math==Math?self:Function("return this")();"number"==typeof __g&&(__g=e)},function(t){t.exports=function(t){try{return!!t()}catch(t){return!0}}},function(t){t.exports=function(t){return"object"==typeof t?null!==t:"function"==typeof t}},function(t,e,n){var r=n(68)("wks"),o=n(44),i=n(5).Symbol,a="function"==typeof i;(t.exports=function(t){return r[t]||(r[t]=a&&i[t]||(a?i:o)("Symbol."+t))}).store=r},function(t,e,n){t.exports=!n(6)(function(){return 7!=Object.defineProperty({},"a",{get:function(){return 7}}).a})},function(t,e,n){var r=n(2),o=n(134),i=n(33),a=Object.defineProperty;e.f=n(9)?Object.defineProperty:function(t,e,n){if(r(t),e=i(e,!0),r(n),o)try{return a(t,e,n)}catch(t){}if("get"in n||"set"in n)throw TypeError("Accessors not supported!");return"value"in n&&(t[e]=n.value),t}},function(t,e,n){var r=n(35),o=Math.min;t.exports=function(t){return t>0?o(r(t),9007199254740991):0}},function(t,e,n){(function(e){if("production"!==e.env.NODE_ENV){var r="function"==typeof Symbol&&Symbol.for&&Symbol.for("react.element")||60103,o=function(t){return"object"==typeof t&&null!==t&&t.$$typeof===r};t.exports=n(403)(o,!0)}else t.exports=n(405)()}).call(e,n(21))},function(t,e,n){var r=n(34);t.exports=function(t){return Object(r(t))}},function(t,e,n){"use strict";(function(e){var n=function(){};"production"!==e.env.NODE_ENV&&(n=function(t,e,n){var r=arguments.length;n=Array(r>2?r-2:0);for(var o=2;r>o;o++)n[o-2]=arguments[o];if(void 0===e)throw Error("` + "`" + `warning(condition, format, ...args)` + "`" + ` requires a warning message argument");if(10>e.length||/^[s\W]*$/.test(e))throw Error("The warning format should be able to uniquely identify this warning. Please, use a more descriptive format than: "+e);if(!t){var i=0,a="Warning: "+e.replace(/%s/g,function(){return n[i++]});try{throw Error(a)}catch(t){}}}),t.exports=n}).call(e,n(21))},function(t){t.exports=function(t){if("function"!=typeof t)throw TypeError(t+" is not a function!");return t}},function(t,e,n){var r=n(407);e.root=r.root,e.branch=r.branch},function(t){var e={}.hasOwnProperty;t.exports=function(t,n){return e.call(t,n)}},function(t,e,n){var r=n(10),o=n(43);t.exports=n(9)?function(t,e,n){return r.f(t,e,o(1,n))}:function(t,e,n){return t[e]=n,t}},function(t,e,n){var r=n(5),o=n(18),i=n(17),a=n(44)("src"),u=Function.toString,c=(""+u).split("toString");n(32).inspectSource=function(t){return u.call(t)},(t.exports=function(t,e,n,u){var s="function"==typeof n;s&&(i(n,"name")||o(n,"name",e)),t[e]!==n&&(s&&(i(n,a)||o(n,a,t[e]?""+t[e]:c.join(e+""))),t===r?t[e]=n:u?t[e]?t[e]=n:o(t,e,n):(delete t[e],o(t,e,n)))})(Function.prototype,"toString",function(){return"function"==typeof this&&this[a]||u.call(this)})},function(t,e,n){var r=n(0),o=n(6),i=n(34),a=/"/g,u=function(t,e,n,r){var o=i(t)+"",u="<"+e;return""!==n&&(u+=" "+n+'="'+(r+"").replace(a,""")+'"'),u+">"+o+""};t.exports=function(t,e){var n={};n[t]=e(u),r(r.P+r.F*o(function(){var e=""[t]('"');return e!==e.toLowerCase()||e.split('"').length>3}),"String",n)}},function(t){function e(){throw Error("setTimeout has not been defined")}function n(){throw Error("clearTimeout has not been defined")}function r(t){if(s===setTimeout)return setTimeout(t,0);if((s===e||!s)&&setTimeout)return s=setTimeout,setTimeout(t,0);try{return s(t,0)}catch(e){try{return s.call(null,t,0)}catch(e){return s.call(this,t,0)}}}function o(t){if(f===clearTimeout)return clearTimeout(t);if((f===n||!f)&&clearTimeout)return f=clearTimeout,clearTimeout(t);try{return f(t)}catch(e){try{return f.call(null,t)}catch(e){return f.call(this,t)}}}function i(){d&&p&&(d=!1,p.length?h=p.concat(h):v=-1,h.length&&a())}function a(){if(!d){var t=r(i);d=!0;for(var e=h.length;e;){for(p=h,h=[];++v1)for(var n=1;arguments.length>n;n++)e[n-1]=arguments[n];h.push(new u(t,e)),1!==h.length||d||r(a)},u.prototype.run=function(){this.fun.apply(null,this.array)},l.title="browser",l.browser=!0,l.env={},l.argv=[],l.version="",l.versions={},l.on=c,l.addListener=c,l.once=c,l.off=c,l.removeListener=c,l.removeAllListeners=c,l.emit=c,l.prependListener=c,l.prependOnceListener=c,l.listeners=function(){return[]},l.binding=function(){throw Error("process.binding is not supported")},l.cwd=function(){return"/"},l.chdir=function(){throw Error("process.chdir is not supported")},l.umask=function(){return 0}},,function(t,e,n){"use strict";Object.defineProperty(e,"__esModule",{value:!0});var r=n(424);n.d(e,"BrowserRouter",function(){return r.a});var o=n(426);n.d(e,"HashRouter",function(){return o.a});var i=n(177);n.d(e,"Link",function(){return i.a});var a=n(428);n.d(e,"MemoryRouter",function(){return a.a});var u=n(430);n.d(e,"NavLink",function(){return u.a});var c=n(433);n.d(e,"Prompt",function(){return c.a});var s=n(434);n.d(e,"Redirect",function(){return s.a});var f=n(179);n.d(e,"Route",function(){return f.a});var l=n(123);n.d(e,"Router",function(){return l.a});var p=n(439);n.d(e,"StaticRouter",function(){return p.a});var h=n(440);n.d(e,"Switch",function(){return h.a});var d=n(441);n.d(e,"matchPath",function(){return d.a});var v=n(442);n.d(e,"withRouter",function(){return v.a})},,function(t,e,n){var r=n(59),o=n(34);t.exports=function(t){return r(o(t))}},function(t,e,n){var r=n(60),o=n(43),i=n(25),a=n(33),u=n(17),c=n(134),s=Object.getOwnPropertyDescriptor;e.f=n(9)?s:function(t,e){if(t=i(t),e=a(e,!0),c)try{return s(t,e)}catch(t){}if(u(t,e))return o(!r.f.call(t,e),t[e])}},function(t,e,n){var r=n(17),o=n(13),i=n(91)("IE_PROTO"),a=Object.prototype;t.exports=Object.getPrototypeOf||function(t){return t=o(t),r(t,i)?t[i]:"function"==typeof t.constructor&&t instanceof t.constructor?t.constructor.prototype:t instanceof Object?a:null}},function(t,e,n){var r=n(15);t.exports=function(t,e,n){if(r(t),void 0===e)return t;switch(n){case 1:return function(n){return t.call(e,n)};case 2:return function(n,r){return t.call(e,n,r)};case 3:return function(n,r,o){return t.call(e,n,r,o)}}return function(){return t.apply(e,arguments)}}},function(t){var e={}.toString;t.exports=function(t){return e.call(t).slice(8,-1)}},function(t,e,n){"use strict";var r=n(6);t.exports=function(t,e){return!!t&&r(function(){e?t.call(null,function(){},1):t.call(null)})}},function(t,e,n){"use strict";(function(e){t.exports=function(t,n,r,o,i,a,u,c){if("production"!==e.env.NODE_ENV&&void 0===n)throw Error("invariant requires an error message argument");if(!t){var s;if(void 0===n)s=Error("Minified exception occurred; use the non-minified dev environment for the full error message and additional helpful warnings.");else{var f=[r,o,i,a,u,c],l=0;s=Error(n.replace(/%s/g,function(){return f[l++]})),s.name="Invariant Violation"}throw s.framesToPop=1,s}}}).call(e,n(21))},function(t){var e=t.exports={version:"2.5.1"};"number"==typeof __e&&(__e=e)},function(t,e,n){var r=n(7);t.exports=function(t,e){if(!r(t))return t;var n,o;if(e&&"function"==typeof(n=t.toString)&&!r(o=n.call(t)))return o;if("function"==typeof(n=t.valueOf)&&!r(o=n.call(t)))return o;if(!e&&"function"==typeof(n=t.toString)&&!r(o=n.call(t)))return o;throw TypeError("Can't convert object to primitive value")}},function(t){t.exports=function(t){if(void 0==t)throw TypeError("Can't call method on "+t);return t}},function(t){var e=Math.ceil,n=Math.floor;t.exports=function(t){return isNaN(t=+t)?0:(t>0?n:e)(t)}},function(t,e,n){var r=n(0),o=n(32),i=n(6);t.exports=function(t,e){var n=(o.Object||{})[t]||Object[t],a={};a[t]=e(n),r(r.S+r.F*i(function(){n(1)}),"Object",a)}},function(t,e,n){var r=n(28),o=n(59),i=n(13),a=n(11),u=n(108);t.exports=function(t,e){var n=1==t,c=2==t,s=3==t,f=4==t,l=6==t,p=5==t||l,h=e||u;return function(e,u,d){for(var v,y,m=i(e),g=o(m),b=r(u,d,3),_=a(g.length),w=0,O=n?h(e,_):c?h(e,0):void 0;_>w;w++)if((p||w in g)&&(v=g[w],y=b(v,w,m),t))if(n)O[w]=y;else if(y)switch(t){case 3:return!0;case 5:return v;case 6:return w;case 2:O.push(v)}else if(f)return!1;return l?-1:s||f?f:O}}},function(t,e,n){"use strict";if(n(9)){var r=n(45),o=n(5),i=n(6),a=n(0),u=n(78),c=n(114),s=n(28),f=n(51),l=n(43),p=n(18),h=n(53),d=n(35),v=n(11),y=n(160),m=n(47),g=n(33),b=n(17),_=n(61),w=n(7),O=n(13),x=n(105),P=n(48),E=n(27),k=n(49).f,j=n(107),S=n(44),T=n(8),C=n(37),M=n(69),N=n(76),R=n(110),A=n(56),L=n(73),I=n(50),F=n(109),D=n(150),U=n(10),W=n(26),B=U.f,z=W.f,q=o.RangeError,V=o.TypeError,G=o.Uint8Array,H=Array.prototype,Y=c.ArrayBuffer,$=c.DataView,K=C(0),J=C(2),Q=C(3),X=C(4),Z=C(5),tt=C(6),et=M(!0),nt=M(!1),rt=R.values,ot=R.keys,it=R.entries,at=H.lastIndexOf,ut=H.reduce,ct=H.reduceRight,st=H.join,ft=H.sort,lt=H.slice,pt=H.toString,ht=H.toLocaleString,dt=T("iterator"),vt=T("toStringTag"),yt=S("typed_constructor"),mt=S("def_constructor"),gt=u.CONSTR,bt=u.TYPED,_t=u.VIEW,wt=C(1,function(t,e){return kt(N(t,t[mt]),e)}),Ot=i(function(){return 1===new G(new Uint16Array([1]).buffer)[0]}),xt=!!G&&!!G.prototype.set&&i(function(){new G(1).set({})}),Pt=function(t,e){var n=d(t);if(0>n||n%e)throw q("Wrong offset!");return n},Et=function(t){if(w(t)&&bt in t)return t;throw V(t+" is not a typed array!")},kt=function(t,e){if(!(w(t)&&yt in t))throw V("It is not a typed array constructor!");return new t(e)},jt=function(t,e){return St(N(t,t[mt]),e)},St=function(t,e){for(var n=0,r=e.length,o=kt(t,r);r>n;)o[n]=e[n++];return o},Tt=function(t,e,n){B(t,e,{get:function(){return this._d[n]}})},Ct=function(t){var e,n,r,o,i,a,u=O(t),c=arguments.length,f=c>1?arguments[1]:void 0,l=void 0!==f,p=j(u);if(void 0!=p&&!x(p)){for(a=p.call(u),r=[],e=0;!(i=a.next()).done;e++)r.push(i.value);u=r}for(l&&c>2&&(f=s(f,arguments[2],2)),e=0,n=v(u.length),o=kt(this,n);n>e;e++)o[e]=l?f(u[e],e):u[e];return o},Mt=function(){for(var t=0,e=arguments.length,n=kt(this,e);e>t;)n[t]=arguments[t++];return n},Nt=!!G&&i(function(){ht.call(new G(1))}),Rt=function(){return ht.apply(Nt?lt.call(Et(this)):Et(this),arguments)},At={copyWithin:function(t,e){return D.call(Et(this),t,e,arguments.length>2?arguments[2]:void 0)},every:function(t){return X(Et(this),t,arguments.length>1?arguments[1]:void 0)},fill:function(){return F.apply(Et(this),arguments)},filter:function(t){return jt(this,J(Et(this),t,arguments.length>1?arguments[1]:void 0))},find:function(t){return Z(Et(this),t,arguments.length>1?arguments[1]:void 0)},findIndex:function(t){return tt(Et(this),t,arguments.length>1?arguments[1]:void 0)},forEach:function(t){K(Et(this),t,arguments.length>1?arguments[1]:void 0)},indexOf:function(t){return nt(Et(this),t,arguments.length>1?arguments[1]:void 0)},includes:function(t){return et(Et(this),t,arguments.length>1?arguments[1]:void 0)},join:function(){return st.apply(Et(this),arguments)},lastIndexOf:function(){return at.apply(Et(this),arguments)},map:function(t){return wt(Et(this),t,arguments.length>1?arguments[1]:void 0)},reduce:function(){return ut.apply(Et(this),arguments)},reduceRight:function(){return ct.apply(Et(this),arguments)},reverse:function(){for(var t,e=this,n=Et(e).length,r=Math.floor(n/2),o=0;r>o;)t=e[o],e[o++]=e[--n],e[n]=t;return e},some:function(t){return Q(Et(this),t,arguments.length>1?arguments[1]:void 0)},sort:function(t){return ft.call(Et(this),t)},subarray:function(t,e){var n=Et(this),r=n.length,o=m(t,r);return new(N(n,n[mt]))(n.buffer,n.byteOffset+o*n.BYTES_PER_ELEMENT,v((void 0===e?r:m(e,r))-o))}},Lt=function(t,e){return jt(this,lt.call(Et(this),t,e))},It=function(t){Et(this);var e=Pt(arguments[1],1),n=this.length,r=O(t),o=v(r.length),i=0;if(o+e>n)throw q("Wrong length!");for(;o>i;)this[e+i]=r[i++]},Ft={entries:function(){return it.call(Et(this))},keys:function(){return ot.call(Et(this))},values:function(){return rt.call(Et(this))}},Dt=function(t,e){return w(t)&&t[bt]&&"symbol"!=typeof e&&e in t&&+e+""==e+""},Ut=function(t,e){return Dt(t,e=g(e,!0))?l(2,t[e]):z(t,e)},Wt=function(t,e,n){return!(Dt(t,e=g(e,!0))&&w(n)&&b(n,"value"))||b(n,"get")||b(n,"set")||n.configurable||b(n,"writable")&&!n.writable||b(n,"enumerable")&&!n.enumerable?B(t,e,n):(t[e]=n.value,t)};gt||(W.f=Ut,U.f=Wt),a(a.S+a.F*!gt,"Object",{getOwnPropertyDescriptor:Ut,defineProperty:Wt}),i(function(){pt.call({})})&&(pt=ht=function(){return st.call(this)});var Bt=h({},At);h(Bt,Ft),p(Bt,dt,Ft.values),h(Bt,{slice:Lt,set:It,constructor:function(){},toString:pt,toLocaleString:Rt}),Tt(Bt,"buffer","b"),Tt(Bt,"byteOffset","o"),Tt(Bt,"byteLength","l"),Tt(Bt,"length","e"),B(Bt,vt,{get:function(){return this[bt]}}),t.exports=function(t,e,n,c){c=!!c;var s=t+(c?"Clamped":"")+"Array",l="get"+t,h="set"+t,d=o[s],m=d||{},g=d&&E(d),b=!d||!u.ABV,O={},x=d&&d.prototype,j=function(t,n){var r=t._d;return r.v[l](n*e+r.o,Ot)},S=function(t,n,r){var o=t._d;c&&(r=0>(r=Math.round(r))?0:r>255?255:255&r),o.v[h](n*e+o.o,r,Ot)},T=function(t,e){B(t,e,{get:function(){return j(this,e)},set:function(t){return S(this,e,t)},enumerable:!0})};b?(d=n(function(t,n,r,o){f(t,d,s,"_d");var i,a,u,c,l=0,h=0;if(w(n)){if(!(n instanceof Y||"ArrayBuffer"==(c=_(n))||"SharedArrayBuffer"==c))return bt in n?St(d,n):Ct.call(d,n);i=n,h=Pt(r,e);var m=n.byteLength;if(void 0===o){if(m%e)throw q("Wrong length!");if(0>(a=m-h))throw q("Wrong length!")}else if((a=v(o)*e)+h>m)throw q("Wrong length!");u=a/e}else u=y(n),a=u*e,i=new Y(a);for(p(t,"_d",{b:i,o:h,l:a,e:u,v:new $(i)});u>l;)T(t,l++)}),x=d.prototype=P(Bt),p(x,"constructor",d)):i(function(){d(1)})&&i(function(){new d(-1)})&&L(function(t){new d,new d(null),new d(1.5),new d(t)},!0)||(d=n(function(t,n,r,o){f(t,d,s);var i;return w(n)?n instanceof Y||"ArrayBuffer"==(i=_(n))||"SharedArrayBuffer"==i?void 0!==o?new m(n,Pt(r,e),o):void 0!==r?new m(n,Pt(r,e)):new m(n):bt in n?St(d,n):Ct.call(d,n):new m(y(n))}),K(g!==Function.prototype?k(m).concat(k(g)):k(m),function(t){t in d||p(d,t,m[t])}),d.prototype=x,r||(x.constructor=d));var C=x[dt],M=!!C&&("values"==C.name||void 0==C.name),N=Ft.values;p(d,yt,!0),p(x,bt,s),p(x,_t,!0),p(x,mt,d),(c?new d(1)[vt]==s:vt in x)||B(x,vt,{get:function(){return s}}),O[s]=d,a(a.G+a.W+a.F*(d!=m),O),a(a.S,s,{BYTES_PER_ELEMENT:e}),a(a.S+a.F*i(function(){m.of.call(d,1)}),s,{from:Ct,of:Mt}),"BYTES_PER_ELEMENT"in x||p(x,"BYTES_PER_ELEMENT",e),a(a.P,s,At),I(s),a(a.P+a.F*xt,s,{set:It}),a(a.P+a.F*!M,s,Ft),r||x.toString==pt||(x.toString=pt),a(a.P+a.F*i(function(){new d(1).slice()}),s,{slice:Lt}),a(a.P+a.F*(i(function(){return[1,2].toLocaleString()!=new d([1,2]).toLocaleString()})||!i(function(){x.toLocaleString.call([1,2])})),s,{toLocaleString:Rt}),A[s]=M?C:N,r||M||p(x,dt,N)}}else t.exports=function(){}},function(t,e,n){var r=n(155),o=n(0),i=n(68)("metadata"),a=i.store||(i.store=new(n(158))),u=function(t,e,n){var o=a.get(t);if(!o){if(!n)return;a.set(t,o=new r)}var i=o.get(e);if(!i){if(!n)return;o.set(e,i=new r)}return i};t.exports={store:a,map:u,has:function(t,e,n){var r=u(e,n,!1);return void 0!==r&&r.has(t)},get:function(t,e,n){var r=u(e,n,!1);return void 0===r?void 0:r.get(t)},set:function(t,e,n,r){u(n,r,!0).set(t,e)},keys:function(t,e){var n=u(t,e,!1),r=[];return n&&n.forEach(function(t,e){r.push(e)}),r},key:function(t){return void 0===t||"symbol"==typeof t?t:t+""},exp:function(t){o(o.S,"Reflect",t)}}},,function(t,e,n){var r=n(44)("meta"),o=n(7),i=n(17),a=n(10).f,u=0,c=Object.isExtensible||function(){return!0},s=!n(6)(function(){return c(Object.preventExtensions({}))}),f=function(t){a(t,r,{value:{i:"O"+ ++u,w:{}}})},l=function(t,e){if(!o(t))return"symbol"==typeof t?t:("string"==typeof t?"S":"P")+t;if(!i(t,r)){if(!c(t))return"F";if(!e)return"E";f(t)}return t[r].i},p=function(t,e){if(!i(t,r)){if(!c(t))return!0;if(!e)return!1;f(t)}return t[r].w},h=function(t){return s&&d.NEED&&c(t)&&!i(t,r)&&f(t),t},d=t.exports={KEY:r,NEED:!1,fastKey:l,getWeak:p,onFreeze:h}},function(t,e,n){var r=n(8)("unscopables"),o=Array.prototype;void 0==o[r]&&n(18)(o,r,{}),t.exports=function(t){o[r][t]=!0}},function(t){t.exports=function(t,e){return{enumerable:!(1&t),configurable:!(2&t),writable:!(4&t),value:e}}},function(t){var e=0,n=Math.random();t.exports=function(t){return"Symbol(".concat(void 0===t?"":t,")_",(++e+n).toString(36))}},function(t){t.exports=!1},function(t,e,n){var r=n(136),o=n(92);t.exports=Object.keys||function(t){return r(t,o)}},function(t,e,n){var r=n(35),o=Math.max,i=Math.min;t.exports=function(t,e){return t=r(t),0>t?o(t+e,0):i(t,e)}},function(t,e,n){var r=n(2),o=n(137),i=n(92),a=n(91)("IE_PROTO"),u=function(){},c=function(){var t,e=n(89)("iframe"),r=i.length;for(e.style.display="none",n(93).appendChild(e),e.src="javascript:",t=e.contentWindow.document,t.open(),t.write(" + +`) diff --git a/web/package-lock.json b/web/package-lock.json new file mode 100644 index 000000000..6ffdc130a --- /dev/null +++ b/web/package-lock.json @@ -0,0 +1,13 @@ +{ + "name": "drone-ui-react", + "version": "1.0.0", + "lockfileVersion": 1, + "requires": true, + "dependencies": { + "yarn": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/yarn/-/yarn-1.6.0.tgz", + "integrity": "sha1-nOxveYbcI3057HBc502VFV/lXUs=" + } + } +} diff --git a/web/package.json b/web/package.json new file mode 100644 index 000000000..7920fff2f --- /dev/null +++ b/web/package.json @@ -0,0 +1,104 @@ +{ + "name": "drone-ui-react", + "version": "1.0.0", + "main": "index.js", + "scripts": { + "prebuild": "rm -rf dist/files", + "build": "cross-env NODE_ENV=production webpack", + "lint": "eslint src/", + "lesshint": "lesshint --config .lesshintrc src/", + "test": "jest", + "start": "webpack-dev-server --progress --hot --inline", + "format": "prettier --use-tabs --trailing-comma all --write {src/*.js,src/**/*.js,src/**/*/*.js,src/*/*/*/*.js,src/*/*/*/*/*.js,src/*/*/*/*/*/*.js,src/*/*/*/*/*/*.js,src/*/*/*/*/*/*/*.js}" + }, + "jest": { + "moduleFileExtensions": [ + "js", + "jsx" + ], + "moduleDirectories": [ + "src", + "node_modules" + ], + "moduleNameMapper": { + "\\.(jpg|jpeg|png|gif|eot|otf|webp|svg|ttf|woff|woff2|mp4|webm|wav|mp3|m4a|aac|oga)$": "/__mocks__/fileMock.js", + "\\.(css|less)$": "identity-obj-proxy", + "^react$": "preact-compat-enzyme", + "^react-dom/server$": "preact-render-to-string", + "^react-dom$": "preact-compat-enzyme", + "^react-addons-test-utils$": "preact-test-utils" + }, + "collectCoverageFrom": [ + "src/**/*.{js,jsx}" + ] + }, + "author": "Brad Rydzewski", + "license": "Apache-2.0", + "dependencies": { + "ansi_up": "^2.0.2", + "babel-polyfill": "^6.23.0", + "baobab": "^2.4.3", + "baobab-react": "^2.1.2", + "classnames": "^2.2.5", + "drone-js": "file:./vendor/drone-js/", + "humanize-duration": "^3.10.1", + "preact": "^8.2.1", + "preact-compat": "^3.16.0", + "query-string": "^5.0.0", + "react-collapsible": "^2.6.0", + "react-router": "^4.1.2", + "react-router-dom": "^4.1.2", + "react-screen-size": "^1.0.1", + "react-timeago": "^3.4.3", + "react-title-component": "^1.0.1", + "react-transition-group": "^1.2.0", + "yarn": "^1.6.0" + }, + "devDependencies": { + "babel-core": "^6.25.0", + "babel-eslint": "^7.2.3", + "babel-jest": "^21.0.0", + "babel-loader": "^7.1.1", + "babel-plugin-transform-decorators-legacy": "^1.3.4", + "babel-plugin-transform-es3-member-expression-literals": "^6.22.0", + "babel-plugin-transform-es3-property-literals": "^6.22.0", + "babel-plugin-transform-react-jsx": "^6.24.1", + "babel-preset-env": "^1.6.0", + "babel-preset-es2015": "^6.24.1", + "babel-preset-react": "^6.24.1", + "babel-preset-stage-0": "^6.24.1", + "cross-env": "^5.0.3", + "css-loader": "^0.28.4", + "dotenv": "^4.0.0", + "enzyme": "^2.9.1", + "eslint": "^4.6.1", + "eslint-config-prettier": "^2.4.0", + "eslint-config-standard": "^10.2.1", + "eslint-plugin-import": "^2.7.0", + "eslint-plugin-jest": "^21.0.2", + "eslint-plugin-node": "^5.1.1", + "eslint-plugin-prettier": "^2.2.0", + "eslint-plugin-promise": "^3.5.0", + "eslint-plugin-react": "^7.3.0", + "eslint-plugin-standard": "^3.0.1", + "file-loader": "^0.11.2", + "html-webpack-plugin": "^2.30.1", + "identity-obj-proxy": "^3.0.0", + "jasmine-expect": "^3.7.1", + "jest": "^21.0.1", + "jsdoc": "^3.5.4", + "less": "^2.7.2", + "less-loader": "^4.0.5", + "lesshint": "^4.1.3", + "preact-compat-enzyme": "^0.2.5", + "preact-render-to-string": "^3.6.3", + "preact-test-utils": "^0.1.3", + "prettier": "^1.6.0", + "sinon": "^3.2.1", + "sinon-chai": "^2.13.0", + "style-loader": "^0.18.2", + "url-loader": "^0.5.9", + "webpack": "^3.4.1", + "webpack-dev-server": "^2.6.1" + } +} diff --git a/web/src/config/client/index.js b/web/src/config/client/index.js new file mode 100644 index 000000000..fdf2f4fb4 --- /dev/null +++ b/web/src/config/client/index.js @@ -0,0 +1,3 @@ +import DroneClient from "drone-js"; + +export default DroneClient.fromWindow(); diff --git a/web/src/config/client/inject.js b/web/src/config/client/inject.js new file mode 100644 index 000000000..d1dc8451e --- /dev/null +++ b/web/src/config/client/inject.js @@ -0,0 +1,36 @@ +import React from "react"; + +export const drone = (client, Component) => { + // @see https://github.com/yannickcr/eslint-plugin-react/issues/512 + // eslint-disable-next-line react/display-name + const component = class extends React.Component { + getChildContext() { + return { + drone: client, + }; + } + + render() { + return ; + } + }; + + component.childContextTypes = { + drone: (props, propName) => {}, + }; + + return component; +}; + +export const inject = Component => { + // @see https://github.com/yannickcr/eslint-plugin-react/issues/512 + // eslint-disable-next-line react/display-name + const component = class extends React.Component { + render() { + this.props.drone = this.context.drone; + return ; + } + }; + + return component; +}; diff --git a/web/src/config/state.js b/web/src/config/state.js new file mode 100644 index 000000000..80ca9b83c --- /dev/null +++ b/web/src/config/state.js @@ -0,0 +1,78 @@ +import Baobab from "baobab"; + +const user = window.DRONE_USER; +const sync = window.DRONE_SYNC; + +const state = { + follow: false, + language: "en-US", + + user: { + data: user, + error: undefined, + loaded: true, + syncing: sync, + }, + + feed: { + loaded: false, + error: undefined, + data: {}, + }, + + repos: { + loaded: false, + error: undefined, + data: {}, + }, + + secrets: { + loaded: false, + error: undefined, + data: {}, + }, + + registry: { + error: undefined, + loaded: false, + data: {}, + }, + + builds: { + loaded: false, + error: undefined, + data: {}, + }, + + logs: { + follow: false, + loading: true, + error: false, + data: {}, + }, + + token: { + value: undefined, + error: undefined, + loading: false, + }, + + message: { + show: false, + text: undefined, + error: false, + }, + + location: { + protocol: window.location.protocol, + host: window.location.host, + }, +}; + +const tree = new Baobab(state); + +if (window) { + window.tree = tree; +} + +export default tree; diff --git a/web/src/index.html b/web/src/index.html new file mode 100644 index 000000000..3e5a764f7 --- /dev/null +++ b/web/src/index.html @@ -0,0 +1,12 @@ + + + + + + + + + + + + diff --git a/web/src/index.js b/web/src/index.js new file mode 100644 index 000000000..8f11a76a2 --- /dev/null +++ b/web/src/index.js @@ -0,0 +1,14 @@ +import "babel-polyfill"; +import React from "react"; +import { render } from "react-dom"; + +let root; + +function init() { + let App = require("./screens/drone").default; + root = render(, document.body, root); +} + +init(); + +if (module.hot) module.hot.accept("./screens/drone", init); diff --git a/web/src/public/favicon.png b/web/src/public/favicon.png new file mode 100644 index 000000000..7c1d45bf6 Binary files /dev/null and b/web/src/public/favicon.png differ diff --git a/web/src/screens/drone.js b/web/src/screens/drone.js new file mode 100644 index 000000000..3997e6134 --- /dev/null +++ b/web/src/screens/drone.js @@ -0,0 +1,52 @@ +import React, { Component } from "react"; + +import { root } from "baobab-react/higher-order"; +import tree from "config/state"; +import client from "config/client"; +import { drone } from "config/client/inject"; +import { LoginForm, LoginError } from "screens/login/screens"; +import Title from "./titles"; +import Layout from "./layout"; +import RedirectRoot from "./redirect"; +import { fetchFeedOnce, subscribeToFeedOnce } from "shared/utils/feed"; + +import { BrowserRouter, Route, Switch } from "react-router-dom"; + +// eslint-disable-next-line no-unused-vars +import styles from "./drone.less"; + +if (module.hot) { + require("preact/devtools"); +} + +class App extends Component { + render() { + return ( + +
+ + <Switch> + <Route path="/" exact={true} component={RedirectRoot} /> + <Route path="/login/form" exact={true} component={LoginForm} /> + <Route path="/login/error" exact={true} component={LoginError} /> + <Route path="/" exact={false} component={Layout} /> + </Switch> + </div> + </BrowserRouter> + ); + } +} + +if (tree.exists(["user", "data"])) { + fetchFeedOnce(tree, client); + subscribeToFeedOnce(tree, client); +} + +client.onerror = error => { + console.error(error); + if (error.status === 401) { + tree.unset(["user", "data"]); + } +}; + +export default root(tree, drone(client, App)); diff --git a/web/src/screens/drone.less b/web/src/screens/drone.less new file mode 100644 index 000000000..0653409b8 --- /dev/null +++ b/web/src/screens/drone.less @@ -0,0 +1,15 @@ +:global { + @import url('https://fonts.googleapis.com/css?family=Roboto+Mono|Roboto:300,400,500'); + + div, + span { + font-family: 'Roboto'; + font-size: 16px; + } + + html, + body { + margin: 0px; + padding: 0px; + } +} diff --git a/web/src/screens/feed/components/index.js b/web/src/screens/feed/components/index.js new file mode 100644 index 000000000..4ad6b4fda --- /dev/null +++ b/web/src/screens/feed/components/index.js @@ -0,0 +1,3 @@ +import { List, Item } from "./list"; + +export { List, Item }; diff --git a/web/src/screens/feed/components/list.js b/web/src/screens/feed/components/list.js new file mode 100644 index 000000000..3efc6f2c5 --- /dev/null +++ b/web/src/screens/feed/components/list.js @@ -0,0 +1,55 @@ +import React, { Component } from "react"; + +import Status from "shared/components/status"; +import BuildTime from "shared/components/build_time"; + +import styles from "./list.less"; + +import { StarIcon } from "shared/components/icons/index"; + +export const List = ({ children }) => ( + <div className={styles.list}>{children}</div> +); + +export class Item extends Component { + constructor(props) { + super(props); + + this.handleFave = this.handleFave.bind(this); + } + + handleFave(e) { + e.preventDefault(); + this.props.onFave(this.props.item.full_name); + } + + render() { + const { item, faved } = this.props; + return ( + <div className={styles.item}> + <div onClick={this.handleFave}> + <StarIcon filled={faved} size={16} className={styles.star} /> + </div> + <div className={styles.header}> + <div className={styles.title}>{item.full_name}</div> + <div className={styles.icon}> + {item.status ? <Status status={item.status} /> : <noscript />} + </div> + </div> + + <div className={styles.body}> + <BuildTime + start={item.started_at || item.created_at} + finish={item.finished_at} + /> + </div> + </div> + ); + } + + shouldComponentUpdate(nextProps, nextState) { + return ( + this.props.item !== nextProps.item || this.props.faved !== nextProps.faved + ); + } +} diff --git a/web/src/screens/feed/components/list.less b/web/src/screens/feed/components/list.less new file mode 100644 index 000000000..df8cc06f0 --- /dev/null +++ b/web/src/screens/feed/components/list.less @@ -0,0 +1,67 @@ +@import '~shared/styles/colors'; +@import '~shared/styles/utils'; + +.list { + a { + border-top: 1px solid @gray-light; + color: @gray-dark; + display: block; + text-decoration: none; + + &:first-of-type { + border-top-width: 0px; + } + } +} + +.item { + display: flex; + flex-direction: column; + padding: 20px; + text-decoration: none; + position: relative; + + .header { + display: flex; + margin-bottom: 10px; + } + + .title { + color: @gray-dark; + flex: 1 1 auto; + font-size: 15px; + line-height: 22px; + max-width: 250px; + padding-right: 20px; + .text-ellipsis + } + + .body div time { + color: @gray-dark; + font-size: 13px; + } + + .body time { + color: @gray-dark; + display: inline-block; + font-size: 13px; + line-height: 22px; + margin: 0px; + padding: 0px; + vertical-align: middle; + } + + .body svg { + fill: @gray-dark; + line-height: 22px; + margin-right: 10px; + vertical-align: middle; + } + + .star { + position: absolute; + bottom: 20px; + right: 20px; + fill: @gray; + } +} diff --git a/web/src/screens/feed/index.js b/web/src/screens/feed/index.js new file mode 100644 index 000000000..db9b866c5 --- /dev/null +++ b/web/src/screens/feed/index.js @@ -0,0 +1,196 @@ +import React, { Component } from "react"; +import { Link } from "react-router-dom"; + +import { compareFeedItem } from "shared/utils/feed"; + +import { branch } from "baobab-react/higher-order"; +import { inject } from "config/client/inject"; + +import DroneIcon from "shared/components/logo"; +import { List, Item } from "./components"; + +import style from "./index.less"; + +import Collapsible from "react-collapsible"; + +const binding = (props, context) => { + return { feed: ["feed"] }; +}; + +@inject +@branch(binding) +export default class Sidebar extends Component { + constructor(props, context) { + super(props, context); + + this.setState({ + starred: JSON.parse(localStorage.getItem("starred") || "[]"), + starredOpen: (localStorage.getItem("starredOpen") || "true") === "true", + reposOpen: (localStorage.getItem("reposOpen") || "true") === "true", + }); + + this.handleFilter = this.handleFilter.bind(this); + this.toggleStarred = this.toggleItem.bind(this, "starredOpen"); + this.toggleAll = this.toggleItem.bind(this, "reposOpen"); + } + + shouldComponentUpdate(nextProps, nextState) { + return ( + this.props.feed !== nextProps.feed || + this.state.filter !== nextState.filter || + this.state.starred.length !== nextState.starred.length + ); + } + + handleFilter(e) { + this.setState({ + filter: e.target.value, + }); + } + + toggleItem = item => { + this.setState(state => { + return { [item]: !state[item] }; + }); + + localStorage.setItem(item, this.state[item]); + }; + + renderFeed = (list, renderStarred) => { + return ( + <div> + <List>{list.map(item => this.renderItem(item, renderStarred))}</List> + </div> + ); + }; + + renderItem = (item, renderStarred) => { + const starred = this.state.starred; + if (renderStarred && !starred.includes(item.full_name)) { + return null; + } + return ( + <Link to={`/${item.full_name}`} key={item.full_name}> + <Item + item={item} + onFave={this.onFave} + faved={starred.includes(item.full_name)} + /> + </Link> + ); + }; + + onFave = fullName => { + if (!this.state.starred.includes(fullName)) { + this.setState(state => { + const list = state.starred.concat(fullName); + return { starred: list }; + }); + } else { + this.setState(state => { + const list = state.starred.filter(v => v !== fullName); + return { starred: list }; + }); + } + + localStorage.setItem("starred", JSON.stringify(this.state.starred)); + }; + + render() { + const { feed } = this.props; + const { filter } = this.state; + + const list = feed.data ? Object.values(feed.data) : []; + + const filterFunc = item => { + return !filter || item.full_name.indexOf(filter) !== -1; + }; + + const filtered = list.filter(filterFunc).sort(compareFeedItem); + const starredOpen = this.state.starredOpen; + const reposOpen = this.state.reposOpen; + return ( + <div className={style.feed}> + {LOGO} + <Collapsible + trigger="Starred" + triggerTagName="div" + transitionTime={200} + open={starredOpen} + onOpen={this.toggleStarred} + onClose={this.toggleStarred} + triggerOpenedClassName={style.Collapsible__trigger} + triggerClassName={style.Collapsible__trigger} + > + {feed.loaded === false ? ( + LOADING + ) : feed.error ? ( + ERROR + ) : list.length === 0 ? ( + EMPTY + ) : ( + this.renderFeed(list, true) + )} + </Collapsible> + <Collapsible + trigger="Repos" + triggerTagName="div" + transitionTime={200} + open={reposOpen} + onOpen={this.toggleAll} + onClose={this.toggleAll} + triggerOpenedClassName={style.Collapsible__trigger} + triggerClassName={style.Collapsible__trigger} + > + <input + type="text" + placeholder="Search …" + onChange={this.handleFilter} + /> + {feed.loaded === false ? ( + LOADING + ) : feed.error ? ( + ERROR + ) : list.length === 0 ? ( + EMPTY + ) : filtered.length > 0 ? ( + this.renderFeed(filtered.sort(compareFeedItem), false) + ) : ( + NO_MATCHES + )} + </Collapsible> + </div> + ); + } +} + +const LOGO = ( + <div className={style.brand}> + <DroneIcon /> + <p> + Woodpecker<br /> + <span> + yes,  + <a + href="https://github.com/laszlocph/drone-oss-08/" + target="_blank" + rel="noopener noreferrer" + > + it's a fork + </a> + </span> + </p> + </div> +); + +const LOADING = <div className={style.message}>Loading</div>; + +const EMPTY = <div className={style.message}>Your build feed is empty</div>; + +const NO_MATCHES = <div className={style.message}>No results found</div>; + +const ERROR = ( + <div className={style.message}> + Oops. It looks like there was a problem loading your feed + </div> +); diff --git a/web/src/screens/feed/index.less b/web/src/screens/feed/index.less new file mode 100644 index 000000000..9d7db6033 --- /dev/null +++ b/web/src/screens/feed/index.less @@ -0,0 +1,70 @@ +@import '~shared/styles/colors'; + +.feed { + width: 300px; + + input { + border: 1px solid @gray-light; + font-size: 15px; + height: 24px; + line-height: 24px; + outline: none; + margin: 20px; + padding: 5px; + width: 250px; + border-radius: 2px; + } + + ::-moz-input-placeholder { + color: @gray; + font-size: 15px; + font-weight: 300; + } + + ::-webkit-input-placeholder { + color: @gray; + font-size: 15px; + font-weight: 300; + } +} + +.message { + color: @gray; + font-size: 15px; + margin-top: 50px; + padding: 20px; + text-align: center; +} + +.brand { + align-items: center; + border-bottom: 1px solid @gray-light; + box-sizing: border-box; + display: flex; + height: 60px; + padding: 0px 10px; + + svg { + fill: @gray-dark; + height: 50px; + position: relative; + top: 5px; + } + + p { + font-size: 18px; + } + + span { + font-size: 13px; + color: @gray-dark + } +} + +.Collapsible__trigger { + background-color: @gray-light; + border-radius: 2px; + display: flex; + padding: 10px 20px; + text-shadow: 0px 1px 2px rgba(0, 0, 0, 0.1); +} diff --git a/web/src/screens/layout.js b/web/src/screens/layout.js new file mode 100644 index 000000000..7754fb069 --- /dev/null +++ b/web/src/screens/layout.js @@ -0,0 +1,227 @@ +import React, { Component } from "react"; +import classnames from "classnames"; +import { Route, Switch, Link } from "react-router-dom"; +import { connectScreenSize } from "react-screen-size"; + +import { branch } from "baobab-react/higher-order"; +import { inject } from "config/client/inject"; + +import MenuIcon from "shared/components/icons/menu"; + +import Feed from "screens/feed"; +import RepoRegistry from "screens/repo/screens/registry"; +import RepoSecrets from "screens/repo/screens/secrets"; +import RepoSettings from "screens/repo/screens/settings"; +import RepoBuilds from "screens/repo/screens/builds"; +import UserRepos, { UserRepoTitle } from "screens/user/screens/repos"; +import UserTokens from "screens/user/screens/tokens"; +import RedirectRoot from "./redirect"; + +import RepoHeader from "screens/repo/screens/builds/header"; + +import UserReposMenu from "screens/user/screens/repos/menu"; +import BuildLogs, { BuildLogsTitle } from "screens/repo/screens/build"; +import BuildMenu from "screens/repo/screens/build/menu"; +import RepoMenu from "screens/repo/screens/builds/menu"; + +import { Snackbar } from "shared/components/snackbar"; +import { Drawer, DOCK_RIGHT } from "shared/components/drawer/drawer"; + +import styles from "./layout.less"; + +const binding = (props, context) => { + return { + user: ["user"], + message: ["message"], + sidebar: ["sidebar"], + menu: ["menu"], + }; +}; + +const mapScreenSizeToProps = screenSize => { + return { + isTablet: screenSize["small"], + isMobile: screenSize["mobile"], + isDesktop: screenSize["> small"], + }; +}; + +@inject +@branch(binding) +@connectScreenSize(mapScreenSizeToProps) +export default class Default extends Component { + constructor(props, context) { + super(props, context); + this.state = { + menu: false, + feed: false, + }; + + this.openMenu = this.openMenu.bind(this); + this.closeMenu = this.closeMenu.bind(this); + this.closeSnackbar = this.closeSnackbar.bind(this); + } + + componentWillReceiveProps(nextProps) { + if (nextProps.location !== this.props.location) { + this.closeMenu(true); + } + } + + openMenu() { + this.props.dispatch(tree => { + tree.set(["menu"], true); + }); + } + + closeMenu() { + this.props.dispatch(tree => { + tree.set(["menu"], false); + }); + } + + render() { + const { user, message, menu } = this.props; + + const classes = classnames(!user || !user.data ? styles.guest : null); + return ( + <div className={classes}> + <div className={styles.left}> + <Switch> + <Route path={"/"} component={Feed} /> + </Switch> + </div> + <div className={styles.center}> + {!user || !user.data ? ( + <a + href={"/login?url=" + window.location.href} + target="_self" + className={styles.login} + > + Click to Login + </a> + ) : ( + <noscript /> + )} + <div className={styles.title}> + <Switch> + <Route path="/account/repos" component={UserRepoTitle} /> + <Route + path="/:owner/:repo/:build(\d*)/:proc(\d*)" + exact={true} + component={BuildLogsTitle} + /> + <Route + path="/:owner/:repo/:build(\d*)" + component={BuildLogsTitle} + /> + <Route path="/:owner/:repo" component={RepoHeader} /> + </Switch> + {user && user.data ? ( + <div className={styles.avatar}> + <img src={user.data.avatar_url} /> + </div> + ) : ( + undefined + )} + {user && user.data ? ( + <button onClick={this.openMenu}> + <MenuIcon /> + </button> + ) : ( + <noscript /> + )} + </div> + + <div className={styles.menu}> + <Switch> + <Route + path="/account/repos" + exact={true} + component={UserReposMenu} + /> + <Route + path="/account/" + exact={false} + component={undefined} + />BuildMenu + <Route + path="/:owner/:repo/:build(\d*)/:proc(\d*)" + exact={true} + component={BuildMenu} + /> + <Route + path="/:owner/:repo/:build(\d*)" + exact={true} + component={BuildMenu} + /> + <Route path="/:owner/:repo" exact={false} component={RepoMenu} /> + </Switch> + </div> + + <Switch> + <Route path="/account/token" exact={true} component={UserTokens} /> + <Route path="/account/repos" exact={true} component={UserRepos} /> + <Route + path="/:owner/:repo/settings/secrets" + exact={true} + component={RepoSecrets} + /> + <Route + path="/:owner/:repo/settings/registry" + exact={true} + component={RepoRegistry} + /> + <Route + path="/:owner/:repo/settings" + exact={true} + component={RepoSettings} + /> + <Route + path="/:owner/:repo/:build(\d*)" + exact={true} + component={BuildLogs} + /> + <Route + path="/:owner/:repo/:build(\d*)/:proc(\d*)" + exact={true} + component={BuildLogs} + /> + <Route path="/:owner/:repo" exact={true} component={RepoBuilds} /> + <Route path="/" exact={true} component={RedirectRoot} /> + </Switch> + </div> + + <Snackbar message={message.text} onClose={this.closeSnackbar} /> + + <Drawer onClick={this.closeMenu} position={DOCK_RIGHT} open={menu}> + <section> + <ul> + <li> + <Link to="/account/repos">Repositories</Link> + </li> + <li> + <Link to="/account/token">Token</Link> + </li> + </ul> + </section> + <section> + <ul> + <li> + <a href="/logout" target="_self"> + Logout + </a> + </li> + </ul> + </section> + </Drawer> + </div> + ); + } + + closeSnackbar() { + this.props.dispatch(tree => { + tree.unset(["message", "text"]); + }); + } +} diff --git a/web/src/screens/layout.less b/web/src/screens/layout.less new file mode 100644 index 000000000..ab444e081 --- /dev/null +++ b/web/src/screens/layout.less @@ -0,0 +1,85 @@ +@import '~shared/styles/colors'; + +.title { + align-items: center; + border-bottom: 1px solid @gray-light; + box-sizing: border-box; + display: flex; + height: 60px; + padding: 0px 20px; + + &> :first-child { + flex: 1; + } + + .avatar { + align-items: center; + display: flex; + + img { + border-radius: 50%; + height: 28px; + width: 28px; + } + } + + button { + align-items: stretch; + background: @white; + border: 0px; + cursor: pointer; + display: flex; + margin: 0px; + margin-left: 10px; + outline: none; + padding: 0px; + } +} + + +.menu {} + +.left { + border-right: 1px solid @splitter-border-color; + bottom: 0px; + box-sizing: border-box; + left: 0px; + overflow: hidden; + overflow-y: auto; + position: fixed; + right: 0px; + top: 0px; + width: 300px; +} + +.center { + box-sizing: border-box; + padding-left: 300px; +} + +.login { + background: @yellow; + box-sizing: border-box; + color: @white; + display: block; + font-size: 15px; + line-height: 50px; + + // HACK + margin-top: -1px; + padding: 0px 30px; + text-align: center; + text-decoration: none; + text-shadow: 0px 1px 2px rgba(0, 0, 0, 0.1); + text-transform: uppercase; +} + +.guest { + .left { + display: none; + } + + .center { + padding-left: 0px; + } +} diff --git a/web/src/screens/login/screens/error/index.js b/web/src/screens/login/screens/error/index.js new file mode 100644 index 000000000..7f398b297 --- /dev/null +++ b/web/src/screens/login/screens/error/index.js @@ -0,0 +1,34 @@ +import React, { Component } from "react"; +import queryString from "query-string"; +import Icon from "shared/components/icons/report"; + +import styles from "./index.less"; + +const DEFAULT_ERROR = "The system failed to process your Login request."; + +class Error extends Component { + render() { + const parsed = queryString.parse(window.location.search); + let error = DEFAULT_ERROR; + + switch (parsed.code || parsed.error) { + case "oauth_error": + break; + case "access_denied": + break; + } + + return ( + <div className={styles.root}> + <div className={styles.alert}> + <div> + <Icon /> + </div> + <div>{error}</div> + </div> + </div> + ); + } +} + +export default Error; diff --git a/web/src/screens/login/screens/error/index.less b/web/src/screens/login/screens/error/index.less new file mode 100644 index 000000000..a1e5944b4 --- /dev/null +++ b/web/src/screens/login/screens/error/index.less @@ -0,0 +1,34 @@ +@import '~shared/styles/colors'; + +@font: 'Roboto'; + +.root { + box-sizing: border-box; + margin: 50px auto; + max-width: 400px; + min-width: 400px; + padding: 30px; + + .alert { + background: @yellow; + color: @white; + display: flex; + margin-bottom: 20px; + padding: 20px; + text-align: left; + + &> :last-child { + font-family: @font; + font-size: 15px; + line-height: 20px; + padding-left: 10px; + padding-top: 2px; + } + } + + svg { + fill: @white; + height: 26px; + width: 26px; + } +} diff --git a/web/src/screens/login/screens/form/index.js b/web/src/screens/login/screens/form/index.js new file mode 100644 index 000000000..eaa970e3c --- /dev/null +++ b/web/src/screens/login/screens/form/index.js @@ -0,0 +1,21 @@ +import React from "react"; + +import styles from "./index.less"; + +const LoginForm = props => ( + <div className={styles.login}> + <form method="post" action="/authorize"> + <p>Login with your version control system username and password.</p> + <input + placeholder="Username" + name="username" + type="text" + spellCheck="false" + /> + <input placeholder="Password" name="password" type="password" /> + <input value="Login" type="submit" /> + </form> + </div> +); + +export default LoginForm; diff --git a/web/src/screens/login/screens/form/index.less b/web/src/screens/login/screens/form/index.less new file mode 100644 index 000000000..394a4ce5c --- /dev/null +++ b/web/src/screens/login/screens/form/index.less @@ -0,0 +1,69 @@ +@import '~shared/styles/colors'; + +@font: 'Roboto'; + +.login { + margin-top: 50px; + + p { + color: @gray-dark; + font-family: @font; + line-height: 22px; + margin: 0px; + margin-bottom: 30px; + padding: 0px; + text-align: center; + user-select: none; + } + + input { + box-sizing: border-box; + display: block; + outline: none; + width: 100%; + + &[type='password'], + &[type='text'] { + background: @white; + border: 1px solid @gray-light; + font-family: @font; + margin-bottom: 20px; + padding: 10px; + + &:focus { + border: 1px solid @gray-dark; + } + } + + &[type='submit'] { + background: @gray-dark; + border: 0px; + color: @white; + font-family: @font; + line-height: 36px; + user-select: none; + } + } + + form { + box-sizing: border-box; + margin: 0px auto; + max-width: 400px; + min-width: 400px; + padding: 30px; + } + + ::-moz-input-placeholder { + color: @gray; + font-size: 16px; + font-weight: 300; + user-select: none; + } + + ::-webkit-input-placeholder { + color: @gray; + font-size: 16px; + font-weight: 300; + user-select: none; + } +} diff --git a/web/src/screens/login/screens/index.js b/web/src/screens/login/screens/index.js new file mode 100644 index 000000000..07b139db4 --- /dev/null +++ b/web/src/screens/login/screens/index.js @@ -0,0 +1,4 @@ +import LoginForm from "./form"; +import LoginError from "./error"; + +export { LoginForm, LoginError }; diff --git a/web/src/screens/redirect.js b/web/src/screens/redirect.js new file mode 100644 index 000000000..a8a01e08c --- /dev/null +++ b/web/src/screens/redirect.js @@ -0,0 +1,41 @@ +import React, { Component } from "react"; +import { Redirect } from "react-router-dom"; +import { branch } from "baobab-react/higher-order"; +import { Message } from "shared/components/sync"; + +const binding = (props, context) => { + return { + feed: ["feed"], + user: ["user", "data"], + syncing: ["user", "syncing"], + }; +}; + +@branch(binding) +export default class RedirectRoot extends Component { + componentWillReceiveProps(nextProps) { + const { user } = nextProps; + if (!user && window) { + window.location.href = "/login?url=" + window.location.href; + } + } + + render() { + const { user, syncing } = this.props; + const { latest, loaded } = this.props.feed; + + return !loaded && syncing ? ( + <Message /> + ) : !loaded ? ( + undefined + ) : !user ? ( + undefined + ) : !latest ? ( + <Redirect to="/account/repos" /> + ) : !latest.number ? ( + <Redirect to={`/${latest.full_name}`} /> + ) : ( + <Redirect to={`/${latest.full_name}/${latest.number}`} /> + ); + } +} diff --git a/web/src/screens/repo/screens/build/components/approval.js b/web/src/screens/repo/screens/build/components/approval.js new file mode 100644 index 000000000..1a19321a9 --- /dev/null +++ b/web/src/screens/repo/screens/build/components/approval.js @@ -0,0 +1,10 @@ +import React from "react"; +import style from "./approval.less"; + +export const Approval = ({ onapprove, ondecline }) => ( + <div className={style.root}> + <p>Pipeline execution is blocked pending administrator approval</p> + <button onClick={onapprove}>Approve</button> + <button onClick={ondecline}>Decline</button> + </div> +); diff --git a/web/src/screens/repo/screens/build/components/approval.less b/web/src/screens/repo/screens/build/components/approval.less new file mode 100644 index 000000000..22321af1c --- /dev/null +++ b/web/src/screens/repo/screens/build/components/approval.less @@ -0,0 +1,34 @@ +@import '~shared/styles/colors'; + +.root { + background: @yellow; + border-radius: 2px; + margin-bottom: 20px; + padding: 20px; + + button { + background: rgba(255, 255, 255, 0.2); + border: 0px; + border-radius: 2px; + color: @white; + cursor: pointer; + font-size: 13px; + line-height: 28px; + margin-right: 10px; + min-width: 100px; + padding: 0px 10px; + text-transform: uppercase; + + &:focus { + border-radius: 2px; + outline: 1px solid @white; + } + } + + p { + color: @white; + font-size: 15px; + margin-bottom: 20px; + margin-top: 0px; + } +} diff --git a/web/src/screens/repo/screens/build/components/details.js b/web/src/screens/repo/screens/build/components/details.js new file mode 100644 index 000000000..c1f05ec1e --- /dev/null +++ b/web/src/screens/repo/screens/build/components/details.js @@ -0,0 +1,42 @@ +import React, { Component } from "react"; + +import BuildMeta from "shared/components/build_event"; +import BuildTime from "shared/components/build_time"; +import { StatusLabel } from "shared/components/status"; + +import styles from "./details.less"; + +export class Details extends Component { + render() { + const { build } = this.props; + + return ( + <div className={styles.info}> + <StatusLabel status={build.status} /> + + <section className={styles.message} style={{ whiteSpace: "pre-line" }}> + {build.message} + </section> + + <section> + <BuildTime + start={build.started_at || build.created_at} + finish={build.finished_at} + /> + </section> + + <section> + <BuildMeta + link={build.link_url} + event={build.event} + commit={build.commit} + branch={build.branch} + target={build.deploy_to} + refspec={build.refspec} + refs={build.ref} + /> + </section> + </div> + ); + } +} diff --git a/web/src/screens/repo/screens/build/components/details.less b/web/src/screens/repo/screens/build/components/details.less new file mode 100644 index 000000000..6d38484b2 --- /dev/null +++ b/web/src/screens/repo/screens/build/components/details.less @@ -0,0 +1,17 @@ +@import '~shared/styles/colors'; + +.info { + section { + border-bottom: 1px solid @gray-light; + font-size: 14px; + line-height: 20px; + margin: 20px 0px; + padding: 0px 10px; + padding-bottom: 20px; + + &:last-of-type { + border-bottom: 0px; + margin-bottom: 0px; + } + } +} diff --git a/web/src/screens/repo/screens/build/components/elapsed.js b/web/src/screens/repo/screens/build/components/elapsed.js new file mode 100644 index 000000000..0d94e5753 --- /dev/null +++ b/web/src/screens/repo/screens/build/components/elapsed.js @@ -0,0 +1,63 @@ +import React, { Component } from "react"; + +export class Elapsed extends Component { + constructor(props, context) { + super(props); + + this.state = { + elapsed: 0, + }; + + this.tick = this.tick.bind(this); + } + + componentDidMount() { + this.timer = setInterval(this.tick, 1000); + } + + componentWillUnmount() { + clearInterval(this.timer); + } + + tick() { + const { start } = this.props; + const stop = ~~(Date.now() / 1000); + this.setState({ + elapsed: stop - start, + }); + } + + render() { + const { elapsed } = this.state; + const date = new Date(null); + date.setSeconds(elapsed); + return ( + <time> + {!elapsed ? ( + undefined + ) : elapsed > 3600 ? ( + date.toISOString().substr(11, 8) + ) : ( + date.toISOString().substr(14, 5) + )} + </time> + ); + } +} + +/* + * Returns the duration in hh:mm:ss format. + * + * @param {number} from - The start time in secnds + * @param {number} to - The end time in seconds + * @return {string} + */ +export const formatTime = (end, start) => { + const diff = end - start; + const date = new Date(null); + date.setSeconds(diff); + + return diff > 3600 + ? date.toISOString().substr(11, 8) + : date.toISOString().substr(14, 5); +}; diff --git a/web/src/screens/repo/screens/build/components/index.js b/web/src/screens/repo/screens/build/components/index.js new file mode 100644 index 000000000..a7fc1de14 --- /dev/null +++ b/web/src/screens/repo/screens/build/components/index.js @@ -0,0 +1,5 @@ +import { Approval } from "./approval"; +import { Details } from "./details"; +import { ProcList, ProcListItem } from "./procs"; + +export { Approval, Details, ProcList, ProcListItem }; diff --git a/web/src/screens/repo/screens/build/components/procs.js b/web/src/screens/repo/screens/build/components/procs.js new file mode 100644 index 000000000..a32392139 --- /dev/null +++ b/web/src/screens/repo/screens/build/components/procs.js @@ -0,0 +1,76 @@ +import React, { Component } from "react"; +import { Link } from "react-router-dom"; +import classnames from "classnames"; + +import { Elapsed, formatTime } from "./elapsed"; +import { default as Status, StatusText } from "shared/components/status"; + +import styles from "./procs.less"; + +const renderEnviron = data => { + return ( + <div> + {data[0]}={data[1]} + </div> + ); +}; + +const ProcListHolder = ({ vars, renderName, children }) => ( + <div className={styles.list}> + {renderName && vars.name !== "drone" ? ( + <div> + <StatusText status={vars.state} text={vars.name} /> + </div> + ) : null} + {vars.environ ? ( + <div> + <StatusText + status={vars.state} + text={Object.entries(vars.environ).map(renderEnviron)} + /> + </div> + ) : null} + {children} + </div> +); + +export class ProcList extends Component { + render() { + const { repo, build, rootProc, selectedProc, renderName } = this.props; + return ( + <ProcListHolder vars={rootProc} renderName={renderName}> + {this.props.rootProc.children.map(function(child) { + return ( + <Link + to={`/${repo.full_name}/${build.number}/${child.pid}`} + key={`${repo.full_name}-${build.number}-${child.pid}`} + > + <ProcListItem + key={child.pid} + name={child.name} + start={child.start_time} + finish={child.end_time} + state={child.state} + selected={child.pid === selectedProc.pid} + /> + </Link> + ); + })} + </ProcListHolder> + ); + } +} + +export const ProcListItem = ({ name, start, finish, state, selected }) => ( + <div className={classnames(styles.item, selected ? styles.selected : null)}> + <h3>{name}</h3> + {finish ? ( + <time>{formatTime(finish, start)}</time> + ) : ( + <Elapsed start={start} /> + )} + <div> + <Status status={state} /> + </div> + </div> +); diff --git a/web/src/screens/repo/screens/build/components/procs.less b/web/src/screens/repo/screens/build/components/procs.less new file mode 100644 index 000000000..033d8cd77 --- /dev/null +++ b/web/src/screens/repo/screens/build/components/procs.less @@ -0,0 +1,49 @@ +@import '~shared/styles/colors'; + +.list { + a { + color: @gray-dark; + display: block; + text-decoration: none; + } +} + +.vars { + padding: 30px 0 0 10px; +} + +.item { + background: @white; + box-sizing: border-box; + display: flex; + padding: 0px 10px; + + &.selected, + &:hover { + background: @gray-light; + } + + time { + color: @gray; + display: inline-block; + font-size: 13px; + line-height: 32px; + margin-right: 15px; + vertical-align: middle; + } + + h3 { + flex: 1 1 auto; + font-size: 14px; + font-weight: normal; + line-height: 36px; + margin: 0px; + padding: 0px; + vertical-align: middle; + } + + &:last-child { + align-items: center; + display: flex; + } +} diff --git a/web/src/screens/repo/screens/build/index.js b/web/src/screens/repo/screens/build/index.js new file mode 100644 index 000000000..13effc7a7 --- /dev/null +++ b/web/src/screens/repo/screens/build/index.js @@ -0,0 +1,257 @@ +import React, { Component } from "react"; +import { Link } from "react-router-dom"; + +import { fetchBuild, approveBuild, declineBuild } from "shared/utils/build"; +import { + STATUS_BLOCKED, + STATUS_DECLINED, + STATUS_ERROR, +} from "shared/constants/status"; + +import { findChildProcess } from "shared/utils/proc"; +import { fetchRepository } from "shared/utils/repository"; + +import Breadcrumb, { SEPARATOR } from "shared/components/breadcrumb"; + +import { Approval, Details, ProcList } from "./components"; + +import { branch } from "baobab-react/higher-order"; +import { inject } from "config/client/inject"; + +import Output from "./logs"; + +import styles from "./index.less"; + +const binding = (props, context) => { + const { owner, repo, build } = props.match.params; + const slug = `${owner}/${repo}`; + const number = parseInt(build); + + return { + repo: ["repos", "data", slug], + build: ["builds", "data", slug, number], + }; +}; + +@inject +@branch(binding) +export default class BuildLogs extends Component { + constructor(props, context) { + super(props, context); + + this.handleApprove = this.handleApprove.bind(this); + this.handleDecline = this.handleDecline.bind(this); + } + + componentWillMount() { + this.synchronize(this.props); + } + + handleApprove() { + const { repo, build, drone } = this.props; + this.props.dispatch( + approveBuild, + drone, + repo.owner, + repo.name, + build.number, + ); + } + + handleDecline() { + const { repo, build, drone } = this.props; + this.props.dispatch( + declineBuild, + drone, + repo.owner, + repo.name, + build.number, + ); + } + + componentWillUpdate(nextProps) { + if (this.props.match.url !== nextProps.match.url) { + this.synchronize(nextProps); + } + } + + synchronize(props) { + if (!props.repo) { + this.props.dispatch( + fetchRepository, + props.drone, + props.match.params.owner, + props.match.params.repo, + ); + } + if (!props.build || !props.build.procs) { + this.props.dispatch( + fetchBuild, + props.drone, + props.match.params.owner, + props.match.params.repo, + props.match.params.build, + ); + } + } + + shouldComponentUpdate(nextProps, nextState) { + return this.props !== nextProps; + } + + render() { + const { repo, build } = this.props; + + if (!build || !repo) { + return this.renderLoading(); + } + + if (build.status === STATUS_DECLINED || build.status === STATUS_ERROR) { + return this.renderError(); + } + + if (build.status === STATUS_BLOCKED) { + return this.renderBlocked(); + } + + if (!build.procs) { + return this.renderLoading(); + } + + return this.renderSimple(); + } + + renderLoading() { + return ( + <div className={styles.host}> + <div className={styles.columns}> + <div className={styles.right}>Loading ...</div> + </div> + </div> + ); + } + + renderBlocked() { + const { build } = this.props; + return ( + <div className={styles.host}> + <div className={styles.columns}> + <div className={styles.right}> + <Details build={build} /> + </div> + <div className={styles.left}> + <Approval + onapprove={this.handleApprove} + ondecline={this.handleDecline} + /> + </div> + </div> + </div> + ); + } + + renderError() { + const { build } = this.props; + return ( + <div className={styles.host}> + <div className={styles.columns}> + <div className={styles.right}> + <Details build={build} /> + </div> + <div className={styles.left}> + <div className={styles.logerror}> + {build.status === STATUS_ERROR ? ( + build.error + ) : ( + "Pipeline execution was declined" + )} + </div> + </div> + </div> + </div> + ); + } + + highlightedLine() { + if (location.hash.startsWith("#L")) { + return parseInt(location.hash.substr(2)) - 1; + } + + return undefined; + } + + renderSimple() { + // if (nextProps.build.procs[0].children !== undefined){ + // return null; + // } + + const { repo, build, match } = this.props; + const selectedProc = match.params.proc + ? findChildProcess(build.procs, match.params.proc) + : build.procs[0].children[0]; + const selectedProcParent = findChildProcess(build.procs, selectedProc.ppid); + const highlighted = this.highlightedLine(); + + return ( + <div className={styles.host}> + <div className={styles.columns}> + <div className={styles.right}> + <Details build={build} /> + <section className={styles.sticky}> + {build.procs.map(function(rootProc) { + return ( + <div style="padding-bottom: 50px;" key={rootProc.pid}> + <ProcList + key={rootProc.pid} + repo={repo} + build={build} + rootProc={rootProc} + selectedProc={selectedProc} + renderName={build.procs.length > 1} + /> + </div> + ); + })} + </section> + </div> + <div className={styles.left}> + {selectedProc && selectedProc.error ? ( + <div className={styles.logerror}>{selectedProc.error}</div> + ) : null} + {selectedProcParent && selectedProcParent.error ? ( + <div className={styles.logerror}>{selectedProcParent.error}</div> + ) : null} + <Output + match={this.props.match} + build={this.props.build} + proc={selectedProc} + highlighted={highlighted} + /> + </div> + </div> + </div> + ); + } +} + +export class BuildLogsTitle extends Component { + render() { + const { owner, repo, build } = this.props.match.params; + return ( + <Breadcrumb + elements={[ + <Link to={`/${owner}/${repo}`} key={`${owner}-${repo}`}> + {owner} / {repo} + </Link>, + SEPARATOR, + <Link + to={`/${owner}/${repo}/${build}`} + key={`${owner}-${repo}-${build}`} + > + {build} + </Link>, + ]} + /> + ); + } +} diff --git a/web/src/screens/repo/screens/build/index.less b/web/src/screens/repo/screens/build/index.less new file mode 100644 index 000000000..fc9f16450 --- /dev/null +++ b/web/src/screens/repo/screens/build/index.less @@ -0,0 +1,51 @@ +@import '~shared/styles/colors'; + +.host { + padding: 0px 20px; + padding-bottom: 20px; + padding-right: 0px; + + .columns { + display: flex; + + .left { + box-sizing: border-box; + flex: 1; + min-width: 0px; + padding-right: 20px; + padding-top: 20px; + } + + .right { + box-sizing: border-box; + flex: 0 0 350px; + min-width: 0px; + padding-right: 20px; + padding-top: 20px; + + &> section { + border-top: 1px solid @gray-light; + padding-top: 20px; + } + } + } +} + +section.sticky { + position: sticky; + top: 0px; + + &:stuck { + border-top-width: 0px; + } +} + +.logerror { + background: @gray-light; + border-radius: 2px; + color: @red; + display: block; + font-size: 14px; + margin-bottom: 10px; + padding: 20px; +} diff --git a/web/src/screens/repo/screens/build/logs/components/anchor.js b/web/src/screens/repo/screens/build/logs/components/anchor.js new file mode 100644 index 000000000..5c2179a93 --- /dev/null +++ b/web/src/screens/repo/screens/build/logs/components/anchor.js @@ -0,0 +1,15 @@ +import React from "react"; + +import styles from "./anchor.less"; + +export const Top = () => <div className={styles.top} />; + +export const Bottom = () => <div className={styles.bottom} />; + +export const scrollToTop = () => { + document.querySelector(`.${styles.top}`).scrollIntoView(); +}; + +export const scrollToBottom = () => { + document.querySelector(`.${styles.bottom}`).scrollIntoView(); +}; diff --git a/web/src/screens/repo/screens/build/logs/components/anchor.less b/web/src/screens/repo/screens/build/logs/components/anchor.less new file mode 100644 index 000000000..27a0c368e --- /dev/null +++ b/web/src/screens/repo/screens/build/logs/components/anchor.less @@ -0,0 +1,4 @@ +.top, +.bottom { + font-size: 0px; +} diff --git a/web/src/screens/repo/screens/build/logs/components/term.js b/web/src/screens/repo/screens/build/logs/components/term.js new file mode 100644 index 000000000..da9e1244e --- /dev/null +++ b/web/src/screens/repo/screens/build/logs/components/term.js @@ -0,0 +1,93 @@ +import React, { Component } from "react"; +import AnsiUp from "ansi_up"; +import style from "./term.less"; +import { Link } from "react-router-dom"; + +let formatter = new AnsiUp(); +formatter.use_classes = true; + +class Term extends Component { + render() { + const { lines, exitcode, highlighted } = this.props; + return ( + <div className={style.term}> + {lines.map(line => renderTermLine(line, highlighted))} + {exitcode !== undefined ? renderExitCode(exitcode) : undefined} + </div> + ); + } + + shouldComponentUpdate(nextProps, nextState) { + return ( + this.props.lines !== nextProps.lines || + this.props.exitcode !== nextProps.exitcode || + this.props.highlighted !== nextProps.highlighted + ); + } +} + +class TermLine extends Component { + render() { + const { line, highlighted } = this.props; + return ( + <div + className={highlighted === line.pos ? style.highlight : style.line} + key={line.pos} + ref={highlighted === line.pos ? ref => (this.ref = ref) : null} + > + <div> + <Link to={`#L${line.pos + 1}`} key={line.pos + 1}> + {line.pos + 1} + </Link> + </div> + <div dangerouslySetInnerHTML={{ __html: this.colored }} /> + <div>{line.time || 0}s</div> + </div> + ); + } + + componentDidMount() { + if (this.ref !== undefined) { + scrollToRef(this.ref); + } + } + + get colored() { + return formatter.ansi_to_html(this.props.line.out || ""); + } + + shouldComponentUpdate(nextProps, nextState) { + return ( + this.props.line.out !== nextProps.line.out || + this.props.highlighted !== nextProps.highlighted + ); + } +} + +const renderTermLine = (line, highlighted) => { + return <TermLine line={line} highlighted={highlighted} />; +}; + +const renderExitCode = code => { + return <div className={style.exitcode}>exit code {code}</div>; +}; + +const TermError = () => { + return ( + <div className={style.error}> + Oops. There was a problem loading the logs. + </div> + ); +}; + +const TermLoading = () => { + return <div className={style.loading}>Loading ...</div>; +}; + +const scrollToRef = ref => window.scrollTo(0, ref.offsetTop - 100); + +Term.Line = TermLine; +Term.Error = TermError; +Term.Loading = TermLoading; + +export default Term; diff --git a/web/src/screens/repo/screens/build/logs/components/term.less b/web/src/screens/repo/screens/build/logs/components/term.less new file mode 100644 index 000000000..cb8c4e83d --- /dev/null +++ b/web/src/screens/repo/screens/build/logs/components/term.less @@ -0,0 +1,86 @@ +@import '~shared/styles/colors'; +@import '~shared/styles/ansi'; + +.term { + background: @gray-light; + border-radius: 2px; + padding: 20px; + + .exitcode { + -moz-user-select: none; + -webkit-user-select: none; + color: rgba(0, 0, 0, 0.3); + font-family: 'Roboto Mono', monospace; + font-size: 13px; + margin-top: 10px; + min-width: 20px; + padding: 0px; + user-select: none; + } +} + + +.line { + color: @gray-dark; + display: flex; + line-height: 19px; + max-width: 100%; + + a, + span, + div { + font-family: 'Roboto Mono', monospace; + font-size: 12px; + } + + a { + text-decoration: none; + color: rgba(0, 0, 0, 0.3); + } + + div:first-child { + -webkit-user-select: none; + color: rgba(0, 0, 0, 0.3); + min-width: 20px; + padding-right: 20px; + user-select: none; + } + + div:nth-child(2) { + flex: 1 1 auto; + min-width: 0px; + white-space: pre-wrap; + word-wrap: break-word; + } + + div:last-child { + -webkit-user-select: none; + color: rgba(0, 0, 0, 0.3); + padding-left: 20px; + user-select: none; + } +} + +.highlight { + .line; + background-color: @yellow; +} + +// log loading message +.loading { + background: @gray-light; + border-radius: 2px; + font-family: 'Roboto Mono', monospace; + font-size: 13px; + padding: 20px; +} + +// log error message +.error { + background: @gray-light; + border-radius: 2px; + color: @red; + font-size: 14px; + margin-bottom: 10px; + padding: 20px; +} diff --git a/web/src/screens/repo/screens/build/logs/index.js b/web/src/screens/repo/screens/build/logs/index.js new file mode 100644 index 000000000..f9ae79926 --- /dev/null +++ b/web/src/screens/repo/screens/build/logs/index.js @@ -0,0 +1,143 @@ +import React, { Component } from "react"; +import { inject } from "config/client/inject"; +import { branch } from "baobab-react/higher-order"; +import { repositorySlug } from "shared/utils/repository"; +import { assertProcFinished, assertProcRunning } from "shared/utils/proc"; +import { fetchLogs, subscribeToLogs, toggleLogs } from "shared/utils/logs"; + +import Term from "./components/term"; + +import { Top, Bottom, scrollToTop, scrollToBottom } from "./components/anchor"; + +import { ExpandIcon, PauseIcon, PlayIcon } from "shared/components/icons/index"; + +import styles from "./index.less"; + +const binding = (props, context) => { + const { owner, repo, build } = props.match.params; + const slug = repositorySlug(owner, repo); + const number = parseInt(build); + const pid = parseInt(props.proc.pid); + + return { + logs: ["logs", "data", slug, number, pid, "data"], + eof: ["logs", "data", slug, number, pid, "eof"], + loading: ["logs", "data", slug, number, pid, "loading"], + error: ["logs", "data", slug, number, pid, "error"], + follow: ["logs", "follow"], + }; +}; + +@inject +@branch(binding) +export default class Output extends Component { + constructor(props, context) { + super(props, context); + this.handleFollow = this.handleFollow.bind(this); + } + + componentWillMount() { + if (this.props.proc) { + this.componentWillUpdate(this.props); + } + } + + componentWillUpdate(nextProps) { + const { loading, logs, eof, error } = nextProps; + const routeChange = this.props.match.url !== nextProps.match.url; + + if (loading || error || (logs && eof)) { + return; + } + + if (assertProcFinished(nextProps.proc)) { + return this.props.dispatch( + fetchLogs, + nextProps.drone, + nextProps.match.params.owner, + nextProps.match.params.repo, + nextProps.build.number, + nextProps.proc.pid, + ); + } + + if (assertProcRunning(nextProps.proc) && (!logs || routeChange)) { + this.props.dispatch( + subscribeToLogs, + nextProps.drone, + nextProps.match.params.owner, + nextProps.match.params.repo, + nextProps.build.number, + nextProps.proc, + ); + } + } + + componentDidUpdate() { + if (this.props.follow) { + scrollToBottom(); + } + } + + handleFollow() { + this.props.dispatch(toggleLogs, !this.props.follow); + } + + render() { + const { logs, error, proc, loading, follow, highlighted } = this.props; + + if (loading || !proc) { + return <Term.Loading />; + } + + if (error) { + return <Term.Error />; + } + + return ( + <div> + <Top /> + <Term + lines={logs || []} + highlighted={highlighted} + exitcode={assertProcFinished(proc) ? proc.exit_code : undefined} + /> + <Bottom /> + <Actions + running={assertProcRunning(proc)} + following={follow} + onfollow={this.handleFollow} + onunfollow={this.handleFollow} + /> + </div> + ); + } +} + +/** + * Component renders floating log actions. These can be used + * to follow, unfollow, scroll to top and scroll to bottom. + */ +const Actions = ({ following, running, onfollow, onunfollow }) => ( + <div className={styles.actions}> + {running && !following ? ( + <button onClick={onfollow} className={styles.follow}> + <PlayIcon /> + </button> + ) : null} + + {running && following ? ( + <button onClick={onunfollow} className={styles.unfollow}> + <PauseIcon /> + </button> + ) : null} + + <button onClick={scrollToTop} className={styles.bottom}> + <ExpandIcon /> + </button> + + <button onClick={scrollToBottom} className={styles.top}> + <ExpandIcon /> + </button> + </div> +); diff --git a/web/src/screens/repo/screens/build/logs/index.less b/web/src/screens/repo/screens/build/logs/index.less new file mode 100644 index 000000000..5ea48accc --- /dev/null +++ b/web/src/screens/repo/screens/build/logs/index.less @@ -0,0 +1,105 @@ +@import '~shared/styles/colors'; + +.loading { + background: @gray-light; + border-radius: 2px; + font-family: 'Roboto Mono', monospace; + font-size: 12px; + padding: 20px; +} + +.error { + background: @gray-light; + border-radius: 2px; + color: @red; + font-size: 14px; + margin-bottom: 10px; + padding: 20px; +} + +.actions { + bottom: 30px; + display: flex; + flex-direction: row; + position: fixed; + right: 30px; + + button { + align-items: center; + background: @white; + border: 1px solid @gray; + color: @gray-dark; + cursor: pointer; + display: flex; + flex-direction: row; + justify-content: center; + margin-left: -1px; + min-height: 32px; + min-width: 32px; + outline: none; + padding: 2px; + + &.bottom svg { + transform: rotate(180deg); + } + + &.follow svg, + &.unfollow svg { + height: 18px; + width: 18px; + } + } + + svg { + fill: @gray-dark; + } +} + +.logactions { + bottom: 30px; + display: flex; + position: fixed; + right: 30px; + + div { + display: flex; + } + + button { + align-items: center; + background: @white; + border: 1px solid @gray-light; + color: @gray-dark; + cursor: pointer; + display: flex; + flex-direction: row; + justify-content: center; + margin-left: -1px; + min-height: 32px; + min-width: 32px; + outline: none; + padding: 2px; + + svg { + fill: @gray-dark; + } + + &.gotoTop { + transform: rotate(180deg); + } + + &.followButton { + svg { + height: 18px; + width: 18px; + } + } + + &.unfollowButton { + svg { + height: 18px; + width: 18px; + } + } + } +} diff --git a/web/src/screens/repo/screens/build/menu.js b/web/src/screens/repo/screens/build/menu.js new file mode 100644 index 000000000..32a865c37 --- /dev/null +++ b/web/src/screens/repo/screens/build/menu.js @@ -0,0 +1,78 @@ +import React, { Component } from "react"; +import RepoMenu from "../builds/menu"; +import { RefreshIcon, CloseIcon } from "shared/components/icons"; + +import { cancelBuild, restartBuild } from "shared/utils/build"; +import { findChildProcess } from "shared/utils/proc"; +import { repositorySlug } from "shared/utils/repository"; + +import { branch } from "baobab-react/higher-order"; +import { inject } from "config/client/inject"; + +const binding = (props, context) => { + const { owner, repo, build } = props.match.params; + const slug = repositorySlug(owner, repo); + const number = parseInt(build); + return { + repo: ["repos", "data", slug], + build: ["builds", "data", slug, number], + }; +}; + +@inject +@branch(binding) +export default class BuildMenu extends Component { + constructor(props, context) { + super(props, context); + + this.handleCancel = this.handleCancel.bind(this); + this.handleRestart = this.handleRestart.bind(this); + } + + handleRestart() { + const { dispatch, drone, repo, build } = this.props; + dispatch(restartBuild, drone, repo.owner, repo.name, build.number); + } + + handleCancel() { + const { dispatch, drone, repo, build, match } = this.props; + const proc = findChildProcess(build.procs, match.params.proc || 2); + + dispatch( + cancelBuild, + drone, + repo.owner, + repo.name, + build.number, + proc.ppid, + ); + } + + render() { + const { build } = this.props; + + const rightSide = !build ? ( + undefined + ) : ( + <section> + {build.status === "pending" || build.status === "running" ? ( + <button onClick={this.handleCancel}> + <CloseIcon /> + <span>Cancel</span> + </button> + ) : ( + <button onClick={this.handleRestart}> + <RefreshIcon /> + <span>Restart Build</span> + </button> + )} + </section> + ); + + return ( + <div> + <RepoMenu {...this.props} right={rightSide} /> + </div> + ); + } +} diff --git a/web/src/screens/repo/screens/builds/components/index.js b/web/src/screens/repo/screens/builds/components/index.js new file mode 100644 index 000000000..4ad6b4fda --- /dev/null +++ b/web/src/screens/repo/screens/builds/components/index.js @@ -0,0 +1,3 @@ +import { List, Item } from "./list"; + +export { List, Item }; diff --git a/web/src/screens/repo/screens/builds/components/list.js b/web/src/screens/repo/screens/builds/components/list.js new file mode 100644 index 000000000..b5a1eaa6f --- /dev/null +++ b/web/src/screens/repo/screens/builds/components/list.js @@ -0,0 +1,55 @@ +import React, { Component } from "react"; + +import Status from "shared/components/status"; +import StatusNumber from "shared/components/status_number"; +import BuildTime from "shared/components/build_time"; +import BuildMeta from "shared/components/build_event"; + +import styles from "./list.less"; + +export const List = ({ children }) => ( + <div className={styles.list}>{children}</div> +); + +export class Item extends Component { + render() { + const { build } = this.props; + return ( + <div className={styles.item}> + <div className={styles.icon}> + <img src={build.author_avatar} /> + </div> + + <div className={styles.body}> + <h3>{build.message.split("\n")[0]}</h3> + </div> + + <div className={styles.meta}> + <BuildMeta + link={build.link_url} + event={build.event} + commit={build.commit} + branch={build.branch} + target={build.deploy_to} + refspec={build.refspec} + refs={build.ref} + /> + </div> + + <div className={styles.break} /> + + <div className={styles.time}> + <BuildTime + start={build.started_at || build.created_at} + finish={build.finished_at} + /> + </div> + + <div className={styles.status}> + <StatusNumber status={build.status} number={build.number} /> + <Status status={build.status} /> + </div> + </div> + ); + } +} diff --git a/web/src/screens/repo/screens/builds/components/list.less b/web/src/screens/repo/screens/builds/components/list.less new file mode 100644 index 000000000..b7e302262 --- /dev/null +++ b/web/src/screens/repo/screens/builds/components/list.less @@ -0,0 +1,163 @@ +@import '~shared/styles/colors'; + +.list { + &> a { + border-bottom: 1px solid @gray-light; + box-sizing: border-box; + color: @gray-dark; + display: block; + padding: 20px 0px; + text-decoration: none; + + &:last-child { + border-bottom: 0px; + } + + a { + // no links inside links + display: none; + } + } +} + +.item { + display: flex; + + .break { + display: none; + } +} + +@media (max-width: 1100px) { + .item { + flex-wrap: wrap; + + .icon { + order: 0px; + } + + .body { + flex: 1; + order: 1; + + h3 { + padding-right: 20px; + } + } + + .meta { + border-left-width: 0px; + margin: 0px; + margin-right: 20px; + margin-top: 20px; + order: 4; + padding: 0px; + padding-left: 52px; + } + + .time { + margin-top: 20px; + order: 5; + } + + .status { + order: 2; + } + + .break { + display: block; + flex-basis: 100%; + height: 0px; + order: 3; + overflow: hidden; + width: 0px; + } + } +} + +.item h3 { + -webkit-box-orient: vertical; + -webkit-line-clamp: 2; + display: -webkit-box; + font-size: 15px; + font-weight: normal; + line-height: 22px; + margin: 0px; + min-height: 22px; + overflow: hidden; +} + +.item em { + font-size: 14px; + font-style: normal; +} + +.item span { + color: @gray; + font-size: 14px; + margin: 0px 5px; +} + +.icon { + margin-left: 10px; + margin-right: 20px; + max-width: 22px; + min-width: 22px; + width: 22px; +} + +.icon img { + border-radius: 50%; + height: 22px; + width: 22px; +} + +.status { + display: inline-block; + text-align: right; + white-space: nowrap; +} + +.status span { + border: 2px solid @green; + border-radius: 2px; + color: @green; + display: inline-block; + line-height: 20px; + margin-right: 10px; + min-width: 65px; + text-align: center; +} + +.status div { + display: inline-block; + vertical-align: middle; + + &:last-child { + margin-left: 20px; + } +} + +.body { + flex: 1; +} + +.meta { + border-left: 1px solid @gray-light; + border-right: 1px solid @gray-light; + box-sizing: border-box; + flex: 0 0 200px; + margin-left: 20px; + margin-right: 20px; + min-width: 200px; + padding-left: 20px; + padding-right: 20px; +} + +.time { + box-sizing: border-box; + flex: 0 0 200px; + margin-right: 20px; + min-width: 200px; + padding-right: 20px; +} diff --git a/web/src/screens/repo/screens/builds/header.js b/web/src/screens/repo/screens/builds/header.js new file mode 100644 index 000000000..eb36f3664 --- /dev/null +++ b/web/src/screens/repo/screens/builds/header.js @@ -0,0 +1,20 @@ +import React, { Component } from "react"; +import { Link } from "react-router-dom"; +import Breadcrumb from "shared/components/breadcrumb"; + +export default class Header extends Component { + render() { + const { owner, repo } = this.props.match.params; + return ( + <div> + <Breadcrumb + elements={[ + <Link to={`/${owner}/${repo}`} key={`${owner}-${repo}`}> + {owner} / {repo} + </Link>, + ]} + /> + </div> + ); + } +} diff --git a/web/src/screens/repo/screens/builds/index.js b/web/src/screens/repo/screens/builds/index.js new file mode 100644 index 000000000..e2aa0008d --- /dev/null +++ b/web/src/screens/repo/screens/builds/index.js @@ -0,0 +1,124 @@ +import React, { Component } from "react"; +import { Link } from "react-router-dom"; +import { List, Item } from "./components"; + +import { fetchBuildList, compareBuild } from "shared/utils/build"; +import { fetchRepository, repositorySlug } from "shared/utils/repository"; + +import { branch } from "baobab-react/higher-order"; +import { inject } from "config/client/inject"; + +import styles from "./index.less"; + +const binding = (props, context) => { + const { owner, repo } = props.match.params; + const slug = repositorySlug(owner, repo); + return { + repo: ["repos", "data", slug], + builds: ["builds", "data", slug], + loaded: ["builds", "loaded"], + error: ["builds", "error"], + }; +}; + +@inject +@branch(binding) +export default class Main extends Component { + constructor(props, context) { + super(props, context); + + this.fetchNextBuildPage = this.fetchNextBuildPage.bind(this); + } + + componentWillMount() { + this.synchronize(this.props); + } + + shouldComponentUpdate(nextProps, nextState) { + return ( + this.props.repo !== nextProps.repo || + (nextProps.builds !== undefined && + this.props.builds !== nextProps.builds) || + this.props.error !== nextProps.error || + this.props.loaded !== nextProps.loaded + ); + } + + componentWillUpdate(nextProps) { + if (this.props.match.url !== nextProps.match.url) { + this.synchronize(nextProps); + } + } + + componentDidUpdate(prevProps) { + if (this.props.location !== prevProps.location) { + window.scrollTo(0, 0); + } + } + + synchronize(props) { + const { drone, dispatch, match, repo } = props; + + if (!repo) { + dispatch(fetchRepository, drone, match.params.owner, match.params.repo); + } + + dispatch(fetchBuildList, drone, match.params.owner, match.params.repo); + } + + fetchNextBuildPage(buildList) { + const { drone, dispatch, match } = this.props; + const page = Math.floor(buildList.length / 50) + 1; + + dispatch( + fetchBuildList, + drone, + match.params.owner, + match.params.repo, + page, + ); + } + + render() { + const { repo, builds, loaded, error } = this.props; + const list = Object.values(builds || {}); + + function renderBuild(build) { + return ( + <Link to={`/${repo.full_name}/${build.number}`} key={build.number}> + <Item build={build} /> + </Link> + ); + } + + if (error) { + return <div>Not Found</div>; + } + + if (!loaded && list.length === 0) { + return <div>Loading</div>; + } + + if (!repo) { + return <div>Loading</div>; + } + + if (list.length === 0) { + return <div>Build list is empty</div>; + } + + return ( + <div className={styles.root}> + <List>{list.sort(compareBuild).map(renderBuild)}</List> + {list.length < repo.last_build && ( + <button + onClick={() => this.fetchNextBuildPage(list)} + className={styles.more} + > + Show more builds + </button> + )} + </div> + ); + } +} diff --git a/web/src/screens/repo/screens/builds/index.less b/web/src/screens/repo/screens/builds/index.less new file mode 100644 index 000000000..ff67b3e32 --- /dev/null +++ b/web/src/screens/repo/screens/builds/index.less @@ -0,0 +1,24 @@ +@import '~shared/styles/colors'; + +.root { + padding: 20px; +} + +button { + background: @white; + border: 1px solid @gray-dark; + border-radius: 2px; + color: @gray-dark; + cursor: pointer; + font-family: 'Roboto'; + font-size: 14px; + line-height: 28px; + outline: none; + padding: 0px 20px; + text-transform: uppercase; + user-select: none; + + &.more { + margin-top: 10px; + } +} diff --git a/web/src/screens/repo/screens/builds/menu.js b/web/src/screens/repo/screens/builds/menu.js new file mode 100644 index 000000000..c1a0ef4ad --- /dev/null +++ b/web/src/screens/repo/screens/builds/menu.js @@ -0,0 +1,15 @@ +import React, { Component } from "react"; +import Menu from "shared/components/menu"; + +export default class RepoMenu extends Component { + render() { + const { owner, repo } = this.props.match.params; + const menu = [ + { to: `/${owner}/${repo}`, label: "Builds" }, + { to: `/${owner}/${repo}/settings/secrets`, label: "Secrets" }, + { to: `/${owner}/${repo}/settings/registry`, label: "Registry" }, + { to: `/${owner}/${repo}/settings`, label: "Settings" }, + ]; + return <Menu items={menu} {...this.props} />; + } +} diff --git a/web/src/screens/repo/screens/builds/menu.less b/web/src/screens/repo/screens/builds/menu.less new file mode 100644 index 000000000..89f279777 --- /dev/null +++ b/web/src/screens/repo/screens/builds/menu.less @@ -0,0 +1,16 @@ +@import '~shared/styles/colors'; + +.root { + border-bottom: 1px solid @gray-light; + box-sizing: border-box; + height: 45px; + line-height: 45px; + padding: 0px 20px; + + a { + color: @gray-dark; + font-size: 15px; + margin-right: 20px; + text-decoration: none; + } +} diff --git a/web/src/screens/repo/screens/registry/components/form.js b/web/src/screens/repo/screens/registry/components/form.js new file mode 100644 index 000000000..4354320f5 --- /dev/null +++ b/web/src/screens/repo/screens/registry/components/form.js @@ -0,0 +1,80 @@ +import React, { Component } from "react"; +import styles from "./form.less"; + +export class Form extends Component { + constructor(props, context) { + super(props, context); + + this.state = { + address: "", + username: "", + password: "", + }; + + this._handleAddressChange = this._handleAddressChange.bind(this); + this._handleUsernameChange = this._handleUsernameChange.bind(this); + this._handlePasswordChange = this._handlePasswordChange.bind(this); + this._handleSubmit = this._handleSubmit.bind(this); + + this.clear = this.clear.bind(this); + } + + _handleAddressChange(event) { + this.setState({ address: event.target.value }); + } + + _handleUsernameChange(event) { + this.setState({ username: event.target.value }); + } + + _handlePasswordChange(event) { + this.setState({ password: event.target.value }); + } + + _handleSubmit() { + const { onsubmit } = this.props; + + const detail = { + address: this.state.address, + username: this.state.username, + password: this.state.password, + }; + + onsubmit({ detail }); + this.clear(); + } + + clear() { + this.setState({ address: "" }); + this.setState({ username: "" }); + this.setState({ password: "" }); + } + + render() { + return ( + <div className={styles.form}> + <input + type="text" + value={this.state.address} + onChange={this._handleAddressChange} + placeholder="Registry Address (e.g. docker.io)" + /> + <input + type="text" + value={this.state.username} + onChange={this._handleUsernameChange} + placeholder="Registry Username" + /> + <textarea + rows="1" + value={this.state.password} + onChange={this._handlePasswordChange} + placeholder="Registry Password" + /> + <div className={styles.actions}> + <button onClick={this._handleSubmit}>Save</button> + </div> + </div> + ); + } +} diff --git a/web/src/screens/repo/screens/registry/components/form.less b/web/src/screens/repo/screens/registry/components/form.less new file mode 100644 index 000000000..d3bd1feeb --- /dev/null +++ b/web/src/screens/repo/screens/registry/components/form.less @@ -0,0 +1,65 @@ +@import '~shared/styles/colors'; + +.form { + input { + border: 1px solid @gray-light; + box-sizing: border-box; + display: block; + margin-bottom: 20px; + outline: none; + padding: 10px; + width: 100%; + + &:focus { + border: 1px solid @gray-dark; + } + } + + textarea { + border: 1px solid @gray-light; + box-sizing: border-box; + display: block; + height: 100px; + margin-bottom: 20px; + outline: none; + padding: 10px; + width: 100%; + + &:focus { + border: 1px solid @gray-dark; + } + } + + .actions { + text-align: right; + } + + button { + background: @white; + border: 1px solid @gray-dark; + border-radius: 2px; + color: @gray-dark; + cursor: pointer; + font-family: 'Roboto'; + font-size: 14px; + line-height: 28px; + outline: none; + padding: 0px 20px; + text-transform: uppercase; + user-select: none; + } + + ::-moz-input-placeholder { + color: @gray; + font-size: 15px; + font-weight: 300; + user-select: none; + } + + ::-webkit-input-placeholder { + color: @gray; + font-size: 15px; + font-weight: 300; + user-select: none; + } +} diff --git a/web/src/screens/repo/screens/registry/components/index.js b/web/src/screens/repo/screens/registry/components/index.js new file mode 100644 index 000000000..ecd6b3500 --- /dev/null +++ b/web/src/screens/repo/screens/registry/components/index.js @@ -0,0 +1,4 @@ +import { Form } from "./form"; +import { List, Item } from "./list"; + +export { Form, List, Item }; diff --git a/web/src/screens/repo/screens/registry/components/list.js b/web/src/screens/repo/screens/registry/components/list.js new file mode 100644 index 000000000..66b8b971c --- /dev/null +++ b/web/src/screens/repo/screens/registry/components/list.js @@ -0,0 +1,15 @@ +import React from "react"; +import styles from "./list.less"; + +export const List = ({ children }) => ( + <div className={styles.list}>{children}</div> +); + +export const Item = props => ( + <div className={styles.item} key={props.name}> + <div>{props.name}</div> + <div> + <button onClick={props.ondelete}>delete</button> + </div> + </div> +); diff --git a/web/src/screens/repo/screens/registry/components/list.less b/web/src/screens/repo/screens/registry/components/list.less new file mode 100644 index 000000000..4609d5b23 --- /dev/null +++ b/web/src/screens/repo/screens/registry/components/list.less @@ -0,0 +1,45 @@ +@import '~shared/styles/colors'; + +.item { + border-bottom: 1px solid @gray-light; + display: flex; + padding: 10px 10px; + padding-bottom: 20px; + + &:last-child { + border-bottom: 0px; + } + + &:first-child { + padding-top: 0px; + } + + &> div:first-child { + flex: 1 1 auto; + font-size: 15px; + line-height: 32px; + text-transform: lowercase; + } + + &> div:last-child { + align-content: stretch; + display: flex; + flex-direction: column; + justify-content: center; + text-align: right; + } + + button { + background: @white; + border: 1px solid @red; + border-radius: 2px; + color: @red; + cursor: pointer; + display: block; + font-size: 13px; + padding: 2px 10px; + text-align: center; + text-decoration: none; + text-transform: uppercase; + } +} diff --git a/web/src/screens/repo/screens/registry/index.js b/web/src/screens/repo/screens/registry/index.js new file mode 100644 index 000000000..d8d78965f --- /dev/null +++ b/web/src/screens/repo/screens/registry/index.js @@ -0,0 +1,103 @@ +import React, { Component } from "react"; + +import { repositorySlug } from "shared/utils/repository"; +import { + fetchRegistryList, + createRegistry, + deleteRegistry, +} from "shared/utils/registry"; + +import { branch } from "baobab-react/higher-order"; +import { inject } from "config/client/inject"; + +import { List, Item, Form } from "./components"; + +import styles from "./index.less"; + +const binding = (props, context) => { + const { owner, repo } = props.match.params; + const slug = repositorySlug(owner, repo); + return { + loaded: ["registry", "loaded"], + registries: ["registry", "data", slug], + }; +}; + +@inject +@branch(binding) +export default class RepoRegistry extends Component { + constructor(props, context) { + super(props, context); + + this.handleDelete = this.handleDelete.bind(this); + this.handleSave = this.handleSave.bind(this); + } + + shouldComponentUpdate(nextProps, nextState) { + return this.props.registries !== nextProps.registries; + } + + componentWillMount() { + const { dispatch, drone, match } = this.props; + const { owner, repo } = match.params; + dispatch(fetchRegistryList, drone, owner, repo); + } + + handleSave(e) { + const { dispatch, drone, match } = this.props; + const { owner, repo } = match.params; + const registry = { + address: e.detail.address, + username: e.detail.username, + password: e.detail.password, + }; + + dispatch(createRegistry, drone, owner, repo, registry); + } + + handleDelete(registry) { + const { dispatch, drone, match } = this.props; + const { owner, repo } = match.params; + dispatch(deleteRegistry, drone, owner, repo, registry.address); + } + + render() { + const { registries, loaded } = this.props; + + if (!loaded) { + return LOADING; + } + + return ( + <div className={styles.root}> + <div className={styles.left}> + {Object.keys(registries || {}).length === 0 ? EMPTY : undefined} + <List> + {Object.values(registries || {}).map(renderRegistry.bind(this))} + </List> + </div> + + <div className={styles.right}> + <Form onsubmit={this.handleSave} /> + </div> + </div> + ); + } +} + +function renderRegistry(registry) { + return ( + <Item + name={registry.address} + ondelete={this.handleDelete.bind(this, registry)} + /> + ); +} + +const LOADING = <div className={styles.loading}>Loading</div>; + +const EMPTY = ( + <div className={styles.empty}> + There are no registry credentials for this repository. + </div> +); diff --git a/web/src/screens/repo/screens/registry/index.less b/web/src/screens/repo/screens/registry/index.less new file mode 100644 index 000000000..630bf47bb --- /dev/null +++ b/web/src/screens/repo/screens/registry/index.less @@ -0,0 +1,34 @@ +@import '~shared/styles/colors'; + +.root { + display: flex; + padding: 20px; +} + +.left { + flex: 1; + margin-right: 20px; +} + +.right { + border-left: 1px solid @gray-light; + flex: 1; + padding-left: 20px; + padding-top: 10px; +} + +@media (max-width: 960px) { + .root { + flex-direction: column; + } + + .list { + margin-right: 0px; + } + + .right { + border-left: 0px; + padding-left: 0px; + padding-top: 20px; + } +} diff --git a/web/src/screens/repo/screens/secrets/components/form.js b/web/src/screens/repo/screens/secrets/components/form.js new file mode 100644 index 000000000..a3e244942 --- /dev/null +++ b/web/src/screens/repo/screens/secrets/components/form.js @@ -0,0 +1,140 @@ +import React, { Component } from "react"; + +import { + EVENT_PUSH, + EVENT_TAG, + EVENT_PULL_REQUEST, + EVENT_DEPLOY, +} from "shared/constants/events"; + +import styles from "./form.less"; + +export class Form extends Component { + constructor(props, context) { + super(props, context); + + this.state = { + name: "", + value: "", + event: [EVENT_PUSH, EVENT_TAG, EVENT_DEPLOY], + }; + + this._handleNameChange = this._handleNameChange.bind(this); + this._handleValueChange = this._handleValueChange.bind(this); + this._handleEventChange = this._handleEventChange.bind(this); + this._handleSubmit = this._handleSubmit.bind(this); + + this.clear = this.clear.bind(this); + } + + _handleNameChange(event) { + this.setState({ name: event.target.value }); + } + + _handleValueChange(event) { + this.setState({ value: event.target.value }); + } + + _handleEventChange(event) { + const selected = this.state.event; + let index; + + if (event.target.checked) { + selected.push(event.target.value); + } else { + index = selected.indexOf(event.target.value); + selected.splice(index, 1); + } + + this.setState({ event: selected }); + } + + _handleSubmit() { + const { onsubmit } = this.props; + + const detail = { + name: this.state.name, + value: this.state.value, + event: this.state.event, + }; + + onsubmit({ detail }); + this.clear(); + } + + clear() { + this.setState({ name: "" }); + this.setState({ value: "" }); + this.setState({ event: [EVENT_PUSH, EVENT_TAG, EVENT_DEPLOY] }); + } + + render() { + let checked = this.state.event.reduce((map, event) => { + map[event] = true; + return map; + }, {}); + + return ( + <div className={styles.form}> + <input + type="text" + name="name" + value={this.state.name} + placeholder="Secret Name" + onChange={this._handleNameChange} + /> + <textarea + rows="1" + name="value" + value={this.state.value} + placeholder="Secret Value" + onChange={this._handleValueChange} + /> + <section> + <h2>Events</h2> + <div> + <label> + <input + type="checkbox" + checked={checked[EVENT_PUSH]} + value={EVENT_PUSH} + onChange={this._handleEventChange} + /> + <span>push</span> + </label> + <label> + <input + type="checkbox" + checked={checked[EVENT_TAG]} + value={EVENT_TAG} + onChange={this._handleEventChange} + /> + <span>tag</span> + </label> + <label> + <input + type="checkbox" + checked={checked[EVENT_PULL_REQUEST]} + value={EVENT_PULL_REQUEST} + onChange={this._handleEventChange} + /> + <span>pull request</span> + </label> + <label> + <input + type="checkbox" + checked={checked[EVENT_DEPLOY]} + value={EVENT_DEPLOY} + onChange={this._handleEventChange} + /> + <span>deploy</span> + </label> + </div> + </section> + <div className={styles.actions}> + <button onClick={this._handleSubmit}>Save</button> + </div> + </div> + ); + } +} diff --git a/web/src/screens/repo/screens/secrets/components/form.less b/web/src/screens/repo/screens/secrets/components/form.less new file mode 100644 index 000000000..a7a9b6832 --- /dev/null +++ b/web/src/screens/repo/screens/secrets/components/form.less @@ -0,0 +1,121 @@ +@import '~shared/styles/colors'; + +.form { + input { + border: 1px solid @gray-light; + box-sizing: border-box; + display: block; + margin-bottom: 20px; + outline: none; + padding: 10px; + width: 100%; + + &:focus { + border: 1px solid @gray-dark; + } + } + + textarea { + border: 1px solid @gray-light; + box-sizing: border-box; + display: block; + height: 100px; + margin-bottom: 20px; + outline: none; + padding: 10px; + width: 100%; + + &:focus { + border: 1px solid @gray-dark; + } + } + + section { + display: flex; + flex: 1 1 auto; + padding-bottom: 20px; + + &> div { + flex: 1; + } + + &:first-child { + padding-top: 0px; + } + + &:last-child { + border-bottom-width: 0px; + } + + @media (max-width: 600px) { + display: flex; + flex-direction: column; + + h2 { + flex: none; + margin-bottom: 20px; + } + + &> :last-child { + padding-left: 20px; + } + } + + h2 { + flex: 0 0 100px; + font-size: 15px; + font-weight: normal; + line-height: 26px; + margin: 0px; + padding: 0px; + } + + label { + display: block; + padding: 0px; + + span { + font-size: 15px; + } + } + + input[type='checkbox'] { + width: initial; + display: inline; + margin: 0px 10px 0px 0px; + } + } + + .actions { + text-align: right; + } + + button { + background: @white; + border: 1px solid @gray-dark; + border-radius: 2px; + color: @gray-dark; + cursor: pointer; + font-family: 'Roboto'; + font-size: 14px; + line-height: 28px; + outline: none; + padding: 0px 20px; + text-transform: uppercase; + user-select: none; + } + + ::-moz-input-placeholder { + color: @gray; + font-size: 15px; + font-weight: 300; + user-select: none; + } + + ::-webkit-input-placeholder { + color: @gray; + font-size: 15px; + font-weight: 300; + user-select: none; + } +} diff --git a/web/src/screens/repo/screens/secrets/components/index.js b/web/src/screens/repo/screens/secrets/components/index.js new file mode 100644 index 000000000..ecd6b3500 --- /dev/null +++ b/web/src/screens/repo/screens/secrets/components/index.js @@ -0,0 +1,4 @@ +import { Form } from "./form"; +import { List, Item } from "./list"; + +export { Form, List, Item }; diff --git a/web/src/screens/repo/screens/secrets/components/list.js b/web/src/screens/repo/screens/secrets/components/list.js new file mode 100644 index 000000000..f2fd4325b --- /dev/null +++ b/web/src/screens/repo/screens/secrets/components/list.js @@ -0,0 +1,20 @@ +import React from "react"; +import styles from "./list.less"; + +export const List = ({ children }) => <div>{children}</div>; + +export const Item = props => ( + <div className={styles.item} key={props.name}> + <div> + {props.name} + <ul>{props.event ? props.event.map(renderEvent) : null}</ul> + </div> + <div> + <button onClick={props.ondelete}>delete</button> + </div> + </div> +); + +const renderEvent = event => { + return <li>{event}</li>; +}; diff --git a/web/src/screens/repo/screens/secrets/components/list.less b/web/src/screens/repo/screens/secrets/components/list.less new file mode 100644 index 000000000..c48237a55 --- /dev/null +++ b/web/src/screens/repo/screens/secrets/components/list.less @@ -0,0 +1,65 @@ +@import '~shared/styles/colors'; + +.item { + border-bottom: 1px solid @gray-light; + display: flex; + padding: 10px 10px; + padding-bottom: 20px; + + &:last-child { + border-bottom: 0px; + } + + &:first-child { + padding-top: 0px; + } + + &> div:first-child { + flex: 1 1 auto; + font-size: 15px; + line-height: 32px; + text-transform: lowercase; + } + + &> div:last-child { + align-content: stretch; + display: flex; + flex-direction: column; + justify-content: center; + text-align: right; + } + + button { + background: @white; + border: 1px solid @red; + border-radius: 2px; + color: @red; + cursor: pointer; + display: block; + font-size: 13px; + padding: 2px 10px; + text-align: center; + text-decoration: none; + text-transform: uppercase; + } + + ul { + line-height: 0px; + list-style: none; + margin: 0px; + padding: 0px; + } + + li { + background: @gray-light; + border-radius: 2px; + color: @gray-dark; + display: inline-block; + font-size: 12px; + line-height: 20px; + margin-bottom: 2px; + margin-right: 2px; + padding: 0px 10px; + text-transform: uppercase; + } +} diff --git a/web/src/screens/repo/screens/secrets/index.js b/web/src/screens/repo/screens/secrets/index.js new file mode 100644 index 000000000..7f0f9c672 --- /dev/null +++ b/web/src/screens/repo/screens/secrets/index.js @@ -0,0 +1,99 @@ +import React, { Component } from "react"; + +import { repositorySlug } from "shared/utils/repository"; +import { + fetchSecretList, + createSecret, + deleteSecret, +} from "shared/utils/secrets"; + +import { branch } from "baobab-react/higher-order"; +import { inject } from "config/client/inject"; + +import { List, Item, Form } from "./components"; + +import styles from "./index.less"; + +const binding = (props, context) => { + const { owner, repo } = props.match.params; + const slug = repositorySlug(owner, repo); + return { + loaded: ["secrets", "loaded"], + secrets: ["secrets", "data", slug], + }; +}; + +@inject +@branch(binding) +export default class RepoSecrets extends Component { + constructor(props, context) { + super(props, context); + + this.handleSave = this.handleSave.bind(this); + } + + shouldComponentUpdate(nextProps, nextState) { + return this.props.secrets !== nextProps.secrets; + } + + componentWillMount() { + const { owner, repo } = this.props.match.params; + this.props.dispatch(fetchSecretList, this.props.drone, owner, repo); + } + + handleSave(e) { + const { dispatch, drone, match } = this.props; + const { owner, repo } = match.params; + const secret = { + name: e.detail.name, + value: e.detail.value, + event: e.detail.event, + }; + + dispatch(createSecret, drone, owner, repo, secret); + } + + handleDelete(secret) { + const { dispatch, drone, match } = this.props; + const { owner, repo } = match.params; + dispatch(deleteSecret, drone, owner, repo, secret.name); + } + + render() { + const { secrets, loaded } = this.props; + + if (!loaded) { + return LOADING; + } + + return ( + <div className={styles.root}> + <div className={styles.left}> + {Object.keys(secrets || {}).length === 0 ? EMPTY : undefined} + <List> + {Object.values(secrets || {}).map(renderSecret.bind(this))} + </List> + </div> + <div className={styles.right}> + <Form onsubmit={this.handleSave} /> + </div> + </div> + ); + } +} + +function renderSecret(secret) { + return ( + <Item + name={secret.name} + event={secret.event} + ondelete={this.handleDelete.bind(this, secret)} + /> + ); +} + +const LOADING = <div className={styles.loading}>Loading</div>; + +const EMPTY = ( + <div className={styles.empty}>There are no secrets for this repository.</div> +); diff --git a/web/src/screens/repo/screens/secrets/index.less b/web/src/screens/repo/screens/secrets/index.less new file mode 100644 index 000000000..630bf47bb --- /dev/null +++ b/web/src/screens/repo/screens/secrets/index.less @@ -0,0 +1,34 @@ +@import '~shared/styles/colors'; + +.root { + display: flex; + padding: 20px; +} + +.left { + flex: 1; + margin-right: 20px; +} + +.right { + border-left: 1px solid @gray-light; + flex: 1; + padding-left: 20px; + padding-top: 10px; +} + +@media (max-width: 960px) { + .root { + flex-direction: column; + } + + .list { + margin-right: 0px; + } + + .right { + border-left: 0px; + padding-left: 0px; + padding-top: 20px; + } +} diff --git a/web/src/screens/repo/screens/settings/index.js b/web/src/screens/repo/screens/settings/index.js new file mode 100644 index 000000000..5bff47b90 --- /dev/null +++ b/web/src/screens/repo/screens/settings/index.js @@ -0,0 +1,244 @@ +import React, { Component } from "react"; + +import { branch } from "baobab-react/higher-order"; +import { inject } from "config/client/inject"; + +import { + fetchRepository, + updateRepository, + repositorySlug, +} from "shared/utils/repository"; + +import { + VISIBILITY_PUBLIC, + VISIBILITY_PRIVATE, + VISIBILITY_INTERNAL, +} from "shared/constants/visibility"; + +import styles from "./index.less"; + +const binding = (props, context) => { + const { owner, repo } = props.match.params; + const slug = repositorySlug(owner, repo); + return { + user: ["user", "data"], + repo: ["repos", "data", slug], + }; +}; + +@inject +@branch(binding) +export default class Settings extends Component { + constructor(props, context) { + super(props, context); + + this.handlePushChange = this.handlePushChange.bind(this); + this.handlePullChange = this.handlePullChange.bind(this); + this.handleTagChange = this.handleTagChange.bind(this); + this.handleDeployChange = this.handleDeployChange.bind(this); + this.handleTrustedChange = this.handleTrustedChange.bind(this); + this.handleProtectedChange = this.handleProtectedChange.bind(this); + this.handleVisibilityChange = this.handleVisibilityChange.bind(this); + this.handleTimeoutChange = this.handleTimeoutChange.bind(this); + this.handlePathChange = this.handlePathChange.bind(this); + this.handleFallbackChange = this.handleFallbackChange.bind(this); + this.handleChange = this.handleChange.bind(this); + } + + shouldComponentUpdate(nextProps, nextState) { + return this.props.repo !== nextProps.repo; + } + + componentWillMount() { + const { drone, dispatch, match, repo } = this.props; + + if (!repo) { + dispatch(fetchRepository, drone, match.params.owner, match.params.repo); + } + } + + render() { + const { repo } = this.props; + + if (!repo) { + return undefined; + } + + return ( + <div className={styles.root}> + <section> + <h2>Pipeline Path</h2> + <div> + <input + type="text" + value={repo.config_file} + onBlur={this.handlePathChange} + /> + <label> + <input + type="checkbox" + checked={repo.fallback} + onChange={this.handleFallbackChange} + /> + <span>Fallback to .drone.yml if path not exists</span> + </label> + </div> + </section> + <section> + <h2>Repository Hooks</h2> + <div> + <label> + <input + type="checkbox" + checked={repo.allow_push} + onChange={this.handlePushChange} + /> + <span>push</span> + </label> + <label> + <input + type="checkbox" + checked={repo.allow_pr} + onChange={this.handlePullChange} + /> + <span>pull request</span> + </label> + <label> + <input + type="checkbox" + checked={repo.allow_tags} + onChange={this.handleTagChange} + /> + <span>tag</span> + </label> + <label> + <input + type="checkbox" + checked={repo.allow_deploys} + onChange={this.handleDeployChange} + /> + <span>deployment</span> + </label> + </div> + </section> + + <section> + <h2>Project Settings</h2> + <div> + <label> + <input + type="checkbox" + checked={repo.gated} + onChange={this.handleProtectedChange} + /> + <span>Protected</span> + </label> + <label> + <input + type="checkbox" + checked={repo.trusted} + onChange={this.handleTrustedChange} + /> + <span>Trusted</span> + </label> + </div> + </section> + + <section> + <h2>Project Visibility</h2> + <div> + <label> + <input + type="radio" + name="visibility" + value="public" + checked={repo.visibility === VISIBILITY_PUBLIC} + onChange={this.handleVisibilityChange} + /> + <span>Public</span> + </label> + <label> + <input + type="radio" + name="visibility" + value="private" + checked={repo.visibility === VISIBILITY_PRIVATE} + onChange={this.handleVisibilityChange} + /> + <span>Private</span> + </label> + <label> + <input + type="radio" + name="visibility" + value="internal" + checked={repo.visibility === VISIBILITY_INTERNAL} + onChange={this.handleVisibilityChange} + /> + <span>Internal</span> + </label> + </div> + </section> + + <section> + <h2>Timeout</h2> + <div> + <input + type="number" + value={repo.timeout} + onBlur={this.handleTimeoutChange} + /> + <span className={styles.minutes}>minutes</span> + </div> + </section> + </div> + ); + } + + handlePushChange(e) { + this.handleChange("allow_push", e.target.checked); + } + + handlePullChange(e) { + this.handleChange("allow_pr", e.target.checked); + } + + handleTagChange(e) { + this.handleChange("allow_tag", e.target.checked); + } + + handleDeployChange(e) { + this.handleChange("allow_deploy", e.target.checked); + } + + handleTrustedChange(e) { + this.handleChange("trusted", e.target.checked); + } + + handleProtectedChange(e) { + this.handleChange("gated", e.target.checked); + } + + handleVisibilityChange(e) { + this.handleChange("visibility", e.target.value); + } + + handleTimeoutChange(e) { + this.handleChange("timeout", parseInt(e.target.value)); + } + + handlePathChange(e) { + this.handleChange("config_file", e.target.value); + } + + handleFallbackChange(e) { + this.handleChange("fallback", e.target.checked); + } + + handleChange(prop, value) { + const { dispatch, drone, repo } = this.props; + let data = {}; + data[prop] = value; + dispatch(updateRepository, drone, repo.owner, repo.name, data); + } +} diff --git a/web/src/screens/repo/screens/settings/index.less b/web/src/screens/repo/screens/settings/index.less new file mode 100644 index 000000000..8a052e374 --- /dev/null +++ b/web/src/screens/repo/screens/settings/index.less @@ -0,0 +1,72 @@ +@import '~shared/styles/colors'; + +.root { + padding: 20px; + + section { + border-bottom: 1px solid @gray-light; + display: flex; + flex: 1 1 auto; + padding: 20px 10px; + + &> div { + flex: 1; + } + + &:first-child { + padding-top: 0px; + } + + &:last-child { + border-bottom-width: 0px; + } + + @media (max-width: 600px) { + display: flex; + flex-direction: column; + + h2 { + flex: none; + margin-bottom: 20px; + } + + &> :last-child { + padding-left: 20px; + } + } + } + + h2 { + flex: 0 0 200px; + font-size: 15px; + font-weight: normal; + line-height: 26px; + margin: 0px; + padding: 0px; + } + + label { + display: block; + padding: 0px; + + span { + font-size: 15px; + } + } + + input[type='checkbox'], + input[type='radio'] { + margin-right: 10px; + } + + input[type='number'] { + border: 1px solid @gray-light; + font-size: 15px; + padding: 5px 10px; + width: 50px; + } + + .minutes { + margin-left: 5px; + } +} diff --git a/web/src/screens/titles.js b/web/src/screens/titles.js new file mode 100644 index 000000000..d37f35986 --- /dev/null +++ b/web/src/screens/titles.js @@ -0,0 +1,29 @@ +import React from "react"; +import { Route, Switch } from "react-router-dom"; +import Title from "react-title-component"; + +// @see https://github.com/yannickcr/eslint-plugin-react/issues/512 +// eslint-disable-next-line react/display-name +export default function() { + return ( + <Switch> + <Route path="/account/tokens" exact={true} component={accountTitle} /> + <Route path="/account/repos" exact={true} component={accountRepos} /> + <Route path="/login" exact={false} component={loginTitle} /> + <Route path="/:owner/:repo" exact={false} component={repoTitle} /> + <Route path="/" exact={false} component={defautTitle} /> + </Switch> + ); +} + +const accountTitle = () => <Title render="Tokens | drone" />; + +const accountRepos = () => <Title render="Repositories | drone" />; + +const loginTitle = () => <Title render="Login | drone" />; + +const repoTitle = ({ match }) => ( + <Title render={`${match.params.owner}/${match.params.repo} | drone`} /> +); + +const defautTitle = () => <Title render="Welcome | drone" />; diff --git a/web/src/screens/user/screens/repos/components/index.js b/web/src/screens/user/screens/repos/components/index.js new file mode 100644 index 000000000..4ad6b4fda --- /dev/null +++ b/web/src/screens/user/screens/repos/components/index.js @@ -0,0 +1,3 @@ +import { List, Item } from "./list"; + +export { List, Item }; diff --git a/web/src/screens/user/screens/repos/components/list.js b/web/src/screens/user/screens/repos/components/list.js new file mode 100644 index 000000000..e14954ddc --- /dev/null +++ b/web/src/screens/user/screens/repos/components/list.js @@ -0,0 +1,40 @@ +import React, { Component } from "react"; +import { Link } from "react-router-dom"; + +import { LaunchIcon } from "shared/components/icons"; +import { Switch } from "./switch"; + +import styles from "./list.less"; + +export const List = ({ children }) => ( + <div className={styles.list}>{children}</div> +); + +export class Item extends Component { + render() { + const { owner, name, active, link, onchange } = this.props; + return ( + <div className={styles.item}> + <div> + {owner}/{name} + </div> + <div className={active ? styles.active : styles.inactive}> + <Link to={link}> + <LaunchIcon /> + </Link> + </div> + <div> + <Switch onchange={onchange} checked={active} /> + </div> + </div> + ); + } + + shouldComponentUpdate(nextProps) { + return ( + this.props.owner !== nextProps.owner || + this.props.name !== nextProps.name || + this.props.active !== nextProps.active + ); + } +} diff --git a/web/src/screens/user/screens/repos/components/list.less b/web/src/screens/user/screens/repos/components/list.less new file mode 100644 index 000000000..9b7200465 --- /dev/null +++ b/web/src/screens/user/screens/repos/components/list.less @@ -0,0 +1,39 @@ +@import '~shared/styles/colors'; + +.item { + border-bottom: 1px solid @gray-light; + display: flex; + padding: 10px 10px; + + &:last-child { + border-bottom-width: 0px; + } + + &> div:first-child { + flex: 1 1 auto; + line-height: 24px; + } + + &> div:nth-child(3) { + align-content: stretch; + display: flex; + flex-direction: column; + justify-content: center; + text-align: right; + } + + a { + margin-right: 20px; + width: 100px; + + svg { + fill: @gray; + height: 20px; + width: 20px; + } + } + + .inactive { + display: none; + } +} diff --git a/web/src/screens/user/screens/repos/components/switch.js b/web/src/screens/user/screens/repos/components/switch.js new file mode 100644 index 000000000..a969f5986 --- /dev/null +++ b/web/src/screens/user/screens/repos/components/switch.js @@ -0,0 +1,13 @@ +import React, { Component } from "react"; +import styles from "./switch.less"; + +export class Switch extends Component { + render() { + const { checked, onchange } = this.props; + return ( + <label className={styles.switch}> + <input type="checkbox" checked={checked} onChange={onchange} /> + </label> + ); + } +} diff --git a/web/src/screens/user/screens/repos/components/switch.less b/web/src/screens/user/screens/repos/components/switch.less new file mode 100644 index 000000000..17bbcd22d --- /dev/null +++ b/web/src/screens/user/screens/repos/components/switch.less @@ -0,0 +1,62 @@ +@import '~shared/styles/colors'; + +.switch { + label { + align-items: center; + cursor: pointer; + display: flex; + margin-bottom: 10px; + } + + input[type='checkbox'] { + -moz-appearance: none; + -ms-appearance: none; + -webkit-appearance: none; + appearance: none; + cursor: pointer; + height: 12px; + margin-right: 30px; + outline: none; + position: relative; + width: 12px; + } + + input[type='checkbox']::before, + input[type='checkbox']::after { + content: ''; + position: absolute; + } + + input[type='checkbox']::before { + background-color: lighten(@gray, 15%); + border-radius: 30px; + height: 100%; + transform: translate(-25%, 0); + transition: all 0.25s ease-in-out; + width: 250%; + } + + input[type='checkbox']::after { + background-color: @gray; + border-radius: 30px; + height: 150%; + margin-left: 10%; + margin-top: -25%; + transform: translate(-60%, 0); + transition: all 0.2s; + width: 150%; + } + + // + // Checked + // + + input[type='checkbox']:checked::after { + background-color: @green; + transform: translate(25%, 0); + } + + input[type='checkbox']:checked::before { + background-color: lighten(@green, 15%); + } +} diff --git a/web/src/screens/user/screens/repos/index.js b/web/src/screens/user/screens/repos/index.js new file mode 100644 index 000000000..917cde929 --- /dev/null +++ b/web/src/screens/user/screens/repos/index.js @@ -0,0 +1,137 @@ +import React, { Component } from "react"; + +import { branch } from "baobab-react/higher-order"; +import { inject } from "config/client/inject"; + +import { + fetchRepostoryList, + disableRepository, + enableRepository, +} from "shared/utils/repository"; + +import { List, Item } from "./components"; +import Breadcrumb, { SEPARATOR } from "shared/components/breadcrumb"; + +import styles from "./index.less"; + +const binding = (props, context) => { + return { + repos: ["repos", "data"], + loaded: ["repos", "loaded"], + error: ["repos", "error"], + }; +}; + +@inject +@branch(binding) +export default class UserRepos extends Component { + constructor(props, context) { + super(props, context); + + this.handleFilter = this.handleFilter.bind(this); + this.renderItem = this.renderItem.bind(this); + this.handleToggle = this.handleToggle.bind(this); + } + + handleFilter(e) { + this.setState({ + search: e.target.value, + }); + } + + handleToggle(repo, e) { + const { dispatch, drone } = this.props; + if (e.target.checked) { + dispatch(enableRepository, drone, repo.owner, repo.name); + } else { + dispatch(disableRepository, drone, repo.owner, repo.name); + } + } + + componentWillMount() { + if (!this._dispatched) { + this._dispatched = true; + this.props.dispatch(fetchRepostoryList, this.props.drone); + } + } + + shouldComponentUpdate(nextProps, nextState) { + return ( + this.props.repos !== nextProps.repos || + this.state.search !== nextState.search + ); + } + + render() { + const { repos, loaded, error } = this.props; + const { search } = this.state; + const list = Object.values(repos || {}); + + if (error) { + return ERROR; + } + + if (!loaded) { + return LOADING; + } + + if (list.length === 0) { + return EMPTY; + } + + const filter = repo => { + return !search || repo.full_name.indexOf(search) !== -1; + }; + + const filtered = list.filter(filter); + + return ( + <div> + <div className={styles.search}> + <input + type="text" + placeholder="Search …" + onChange={this.handleFilter} + /> + </div> + <div className={styles.root}> + {filtered.length === 0 ? NO_MATCHES : null} + <List>{list.filter(filter).map(this.renderItem)}</List> + </div> + </div> + ); + } + + renderItem(repo) { + return ( + <Item + key={repo.full_name} + owner={repo.owner} + name={repo.name} + active={repo.active} + link={`/${repo.full_name}`} + onchange={this.handleToggle.bind(this, repo)} + /> + ); + } +} + +const LOADING = <div>Loading</div>; + +const EMPTY = <div>Your repository list is empty</div>; + +const NO_MATCHES = <div>No matches found</div>; + +const ERROR = <div>Error</div>; + +/* eslint-disable react/jsx-key */ +export class UserRepoTitle extends Component { + render() { + return ( + <Breadcrumb + elements={[<span>Account</span>, SEPARATOR, <span>Repositories</span>]} + /> + ); + } +} +/* eslint-enable react/jsx-key */ diff --git a/web/src/screens/user/screens/repos/index.less b/web/src/screens/user/screens/repos/index.less new file mode 100644 index 000000000..6e7bcbad1 --- /dev/null +++ b/web/src/screens/user/screens/repos/index.less @@ -0,0 +1,31 @@ +@import '~shared/styles/colors'; + +.root { + padding: 20px; +} + +.search { + input { + border: 0px; + border-bottom: 1px solid @gray-light; + box-sizing: border-box; + font-size: 15px; + height: 45px; + line-height: 24px; + outline: none; + padding: 0px 20px; + width: 100%; + } + + ::-moz-input-placeholder { + color: @gray; + font-size: 15px; + font-weight: 300; + } + + ::-webkit-input-placeholder { + color: @gray; + font-size: 15px; + font-weight: 300; + } +} diff --git a/web/src/screens/user/screens/repos/menu.js b/web/src/screens/user/screens/repos/menu.js new file mode 100644 index 000000000..5613c83b8 --- /dev/null +++ b/web/src/screens/user/screens/repos/menu.js @@ -0,0 +1,41 @@ +import React, { Component } from "react"; +import { syncRepostoryList } from "shared/utils/repository"; +import { branch } from "baobab-react/higher-order"; +import { inject } from "config/client/inject"; +import { SyncIcon } from "shared/components/icons"; +import Menu from "shared/components/menu"; + +const binding = (props, context) => { + return { + repos: ["repos"], + }; +}; + +@inject +@branch(binding) +export default class UserReposMenu extends Component { + constructor(props, context) { + super(props, context); + + this.handleClick = this.handleClick.bind(this); + } + + handleClick() { + const { dispatch, drone } = this.props; + dispatch(syncRepostoryList, drone); + } + + render() { + const { loaded } = this.props.repos; + const right = ( + <section> + <button disabled={!loaded} onClick={this.handleClick}> + <SyncIcon /> + <span>Synchronize</span> + </button> + </section> + ); + + return <Menu items={[]} right={right} />; + } +} diff --git a/web/src/screens/user/screens/tokens/index.js b/web/src/screens/user/screens/tokens/index.js new file mode 100644 index 000000000..4477d7d1b --- /dev/null +++ b/web/src/screens/user/screens/tokens/index.js @@ -0,0 +1,59 @@ +import React, { Component } from "react"; + +import { generateToken } from "shared/utils/users"; +import { branch } from "baobab-react/higher-order"; +import { inject } from "config/client/inject"; +import styles from "./index.less"; + +const binding = (props, context) => { + return { + location: ["location"], + token: ["token"], + }; +}; + +@inject +@branch(binding) +export default class Tokens extends Component { + shouldComponentUpdate(nextProps, nextState) { + return ( + this.props.location !== nextProps.location || + this.props.token !== nextProps.token + ); + } + + componentWillMount() { + const { drone, dispatch } = this.props; + + dispatch(generateToken, drone); + } + + render() { + const { location, token } = this.props; + + if (!location || !token) { + return <div>Loading</div>; + } + return ( + <div className={styles.root}> + <h2>Your Personal Token:</h2> + <pre>{token}</pre> + <h2>Example API Usage:</h2> + <pre>{usageWithCURL(location, token)}</pre> + <h2>Example CLI Usage:</h2> + <pre>{usageWithCLI(location, token)}</pre> + </div> + ); + } +} + +const usageWithCURL = (location, token) => { + return `curl -i ${location.protocol}//${location.host}/api/user -H "Authorization: Bearer ${token}"`; +}; + +const usageWithCLI = (location, token) => { + return `export DRONE_SERVER=${location.protocol}//${location.host} + export DRONE_TOKEN=${token} + + drone info`; +}; diff --git a/web/src/screens/user/screens/tokens/index.less b/web/src/screens/user/screens/tokens/index.less new file mode 100644 index 000000000..b0b6b5b0e --- /dev/null +++ b/web/src/screens/user/screens/tokens/index.less @@ -0,0 +1,25 @@ +@import '~shared/styles/colors'; + +.root { + padding: 20px; + + pre { + background: @gray-light; + font-family: 'Roboto Mono', monospace; + font-size: 12px; + margin-bottom: 40px; + max-width: 650px; + padding: 20px; + white-space: pre-line; + word-wrap: break-word; + } + + h2 { + font-size: 15px; + font-weight: normal; + + &:first-of-type { + margin-top: 0px; + } + } +} diff --git a/web/src/shared/components/__tests__/status.test.js b/web/src/shared/components/__tests__/status.test.js new file mode 100644 index 000000000..ccf7d609f --- /dev/null +++ b/web/src/shared/components/__tests__/status.test.js @@ -0,0 +1,32 @@ +import React from "react"; +import { mount } from "enzyme"; + +import Status from "../status"; +import { + STATUS_FAILURE, + STATUS_RUNNING, + STATUS_SUCCESS, +} from "shared/constants/status"; + +jest.dontMock("../status"); + +describe("Status component", () => { + test("updates on status change", () => { + const status = mount(<Status status={STATUS_FAILURE} />); + const instance = status.instance(); + + expect( + instance.shouldComponentUpdate({ status: STATUS_FAILURE }), + ).toBeFalsy(); + expect( + instance.shouldComponentUpdate({ status: STATUS_SUCCESS }), + ).toBeTruthy(); + expect(status.hasClass("failure")).toBeTruthy(); + }); + + test("uses the status as the class name", () => { + const status = mount(<Status status={STATUS_RUNNING} />); + + expect(status.hasClass("running")).toBeTruthy(); + }); +}); diff --git a/web/src/shared/components/avatar.js b/web/src/shared/components/avatar.js new file mode 100644 index 000000000..98b225eb3 --- /dev/null +++ b/web/src/shared/components/avatar.js @@ -0,0 +1,12 @@ +import React, { Component } from "react"; +import styles from "./avatar.less"; + +export default class Avatar extends Component { + render() { + const image = this.props.image; + const style = { + backgroundImage: `url(${image})`, + }; + return <div className={styles.avatar} style={style} />; + } +} diff --git a/web/src/shared/components/avatar.less b/web/src/shared/components/avatar.less new file mode 100644 index 000000000..073cceb0e --- /dev/null +++ b/web/src/shared/components/avatar.less @@ -0,0 +1,15 @@ +.avatar { + align-items: center; + display: flex; + + img { + border-radius: 50%; + height: 32px; + width: 32px; + } + + &.small img { + height: 28px; + width: 28px; + } +} diff --git a/web/src/shared/components/breadcrumb.js b/web/src/shared/components/breadcrumb.js new file mode 100644 index 000000000..d6b314a06 --- /dev/null +++ b/web/src/shared/components/breadcrumb.js @@ -0,0 +1,21 @@ +import React, { Component } from "react"; +import { ExpandIcon, BackIcon } from "shared/components/icons/index"; +import style from "./breadcrumb.less"; + +// breadcrumb separater icon. +export const SEPARATOR = <ExpandIcon size={18} className={style.separator} />; + +// breadcrumb back button. +export const BACK_BUTTON = <BackIcon size={18} className={style.back} />; + +// helper function to render a list item. +const renderItem = (element, index) => { + return <li key={index}>{element}</li>; +}; + +export default class Breadcrumb extends Component { + render() { + const { elements } = this.props; + return <ol className={style.breadcrumb}>{elements.map(renderItem)}</ol>; + } +} diff --git a/web/src/shared/components/breadcrumb.less b/web/src/shared/components/breadcrumb.less new file mode 100644 index 000000000..36aa1bae6 --- /dev/null +++ b/web/src/shared/components/breadcrumb.less @@ -0,0 +1,38 @@ +@import '~shared/styles/colors'; + +.breadcrumb { + display: inline-block; + margin: 0px; + padding: 0px; + text-align: left; + + li { + display: inline-block; + vertical-align: middle; + } + + li > span, + li > div, + a, + a:visited, + a:active { + color: @gray-dark; + font-size: 20px; + text-decoration: none; + } + + svg { + height: 24px; + vertical-align: middle; + width: 24px; + } + + .svg.separator { + margin: 0px 5px; + transform: rotate(270deg); + } + + .svg.back { + margin-right: 20px; + } +} diff --git a/web/src/shared/components/build_event.js b/web/src/shared/components/build_event.js new file mode 100644 index 000000000..3bff114ab --- /dev/null +++ b/web/src/shared/components/build_event.js @@ -0,0 +1,73 @@ +import React, { Component } from "react"; +import { + BranchIcon, + CommitIcon, + DeployIcon, + LaunchIcon, + MergeIcon, + TagIcon, +} from "shared/components/icons/index"; +import { + EVENT_TAG, + EVENT_PULL_REQUEST, + EVENT_DEPLOY, +} from "shared/constants/events"; + +import styles from "./build_event.less"; + +export default class BuildEvent extends Component { + render() { + const { event, branch, commit, refs, refspec, link, target } = this.props; + + return ( + <div className={styles.host}> + <div className={styles.row}> + <div> + <CommitIcon /> + </div> + <div>{commit && commit.substr(0, 10)}</div> + </div> + <div className={styles.row}> + <div> + {event === EVENT_TAG ? ( + <TagIcon /> + ) : event === EVENT_PULL_REQUEST ? ( + <MergeIcon /> + ) : event === EVENT_DEPLOY ? ( + <DeployIcon /> + ) : ( + <BranchIcon /> + )} + </div> + <div> + {event === EVENT_TAG && refs ? ( + trimTagRef(refs) + ) : event === EVENT_PULL_REQUEST && refspec ? ( + trimMergeRef(refs) + ) : event === EVENT_DEPLOY && target ? ( + target + ) : ( + branch + )} + </div> + </div> + <a href={link} target="_blank"> + <LaunchIcon /> + </a> + </div> + ); + } +} + +const trimMergeRef = ref => { + return ref.match(/\d/g) || ref; +}; + +const trimTagRef = ref => { + return ref.startsWith("refs/tags/") ? ref.substr(10) : ref; +}; + +// push +// pull request (ref) +// tag (ref) +// deploy diff --git a/web/src/shared/components/build_event.less b/web/src/shared/components/build_event.less new file mode 100644 index 000000000..b87ea73d7 --- /dev/null +++ b/web/src/shared/components/build_event.less @@ -0,0 +1,34 @@ +@import '~shared/styles/utils'; + +.host { + position: relative; + + svg { + height: 18px; + width: 18px; + } + + a { + display: block; + position: absolute; + right: 0px; + top: 0px; + } +} + +.row { + display: flex; + + :first-child { + align-items: center; + display: flex; + margin-right: 5px; + } + + :last-child { + flex: 1; + font-size: 14px; + line-height: 24px; + .text-ellipsis + } +} diff --git a/web/src/shared/components/build_time.js b/web/src/shared/components/build_time.js new file mode 100644 index 000000000..40e4551e0 --- /dev/null +++ b/web/src/shared/components/build_time.js @@ -0,0 +1,37 @@ +import React, { Component } from "react"; +import { ScheduleIcon, TimelapseIcon } from "shared/components/icons/index"; + +import TimeAgo from "react-timeago"; +import Duration from "./duration"; + +import styles from "./build_time.less"; + +export default class Runtime extends Component { + render() { + const { start, finish } = this.props; + return ( + <div className={styles.host}> + <div className={styles.row}> + <div> + <ScheduleIcon /> + </div> + <div>{start ? <TimeAgo date={start * 1000} /> : <span>--</span>}</div> + </div> + <div className={styles.row}> + <div> + <TimelapseIcon /> + </div> + <div> + {finish ? ( + <Duration start={start} finished={finish} /> + ) : start ? ( + <TimeAgo date={start * 1000} /> + ) : ( + <span>--</span> + )} + </div> + </div> + </div> + ); + } +} diff --git a/web/src/shared/components/build_time.less b/web/src/shared/components/build_time.less new file mode 100644 index 000000000..f6dc3343f --- /dev/null +++ b/web/src/shared/components/build_time.less @@ -0,0 +1,23 @@ +.host { + svg { + height: 16px; + width: 16px; + } +} + +.row { + display: flex; + + :first-child { + align-items: center; + display: flex; + margin-right: 5px; + } + + :last-child { + flex: 1; + font-size: 14px; + line-height: 24px; + white-space: nowrap; + } +} diff --git a/web/src/shared/components/drawer/drawer.js b/web/src/shared/components/drawer/drawer.js new file mode 100644 index 000000000..75670a071 --- /dev/null +++ b/web/src/shared/components/drawer/drawer.js @@ -0,0 +1,62 @@ +import React, { Component } from "react"; +import CloseIcon from "shared/components/icons/close"; +import styles from "./drawer.less"; +import { CSSTransitionGroup } from "react-transition-group"; + +export const DOCK_LEFT = styles.left; +export const DOCK_RIGHT = styles.right; + +export class Drawer extends Component { + render() { + const { open, position } = this.props; + + let classes = [styles.drawer]; + if (open) { + classes.push(styles.open); + } + if (position) { + classes.push(position); + } + + var child = open ? ( + <div key={0} onClick={this.props.onClick} className={styles.backdrop} /> + ) : null; + + return ( + <div className={classes.join(" ")}> + <CSSTransitionGroup + transitionName="fade" + transitionEnterTimeout={150} + transitionLeaveTimeout={150} + transitionAppearTimeout={150} + transitionAppear={true} + transitionEnter={true} + transitionLeave={true} + > + {child} + </CSSTransitionGroup> + <div className={styles.inner}>{this.props.children}</div> + </div> + ); + } +} + +export class CloseButton extends Component { + render() { + return ( + <button className={styles.close} onClick={this.props.onClick}> + <CloseIcon /> + </button> + ); + } +} + +export class MenuButton extends Component { + render() { + return ( + <button className={styles.close} onClick={this.props.onClick}> + Show Menu + </button> + ); + } +} diff --git a/web/src/shared/components/drawer/drawer.less b/web/src/shared/components/drawer/drawer.less new file mode 100644 index 000000000..cc0cd0740 --- /dev/null +++ b/web/src/shared/components/drawer/drawer.less @@ -0,0 +1,182 @@ +@import '~shared/styles/colors'; + +// +// backdrop +// + +.backdrop { + background-color: rgba(0, 0, 0, 0.54); + bottom: 0px; + left: 0px; + position: fixed; + right: 0px; + top: 0px; +} + +// +// drawer wrapper +// + +.inner { + background: @white; + bottom: 0px; + box-shadow: 0px 8px 10px -5px rgba(0, 0, 0, 0.2), 0px 16px 24px 2px rgba(0, 0, 0, 0.14), 0px 6px + 30px 5px rgba(0, 0, 0, 0.12); + box-sizing: border-box; + display: flex; + flex-direction: column; + left: 0px; + overflow: hidden; + position: fixed; + right: 0px; + top: 0px; + transition: left ease-in 0.15s; + width: 300px; +} + +// +// drawer +// + +.drawer { + display: none; + height: 0px; + left: -1000px; + position: fixed; + top: -1000px; + width: 0px; + + &.open { + display: flex; + + .inner { + left: 0px; + transition: left ease-in 0.15s; + } + } + + &.right { + .inner { + left: auto; + right: 0px; + } + } +} + +// +// close button +// + +.close { + align-items: center; + background: transparent; + border: 0px; + cursor: pointer; + display: flex; + margin: 0px; + outline: none; + padding: 10px 10px; + text-align: right; + width: 100%; + + svg { + fill: @gray-light; + } +} + +.right .close { + flex-direction: row-reverse; +} + +// +// menu +// + +.drawer ul { + border-top: 1px solid @gray-light; + margin: 0px; + padding: 10px 0px; + + li { + display: block; + margin: 0px; + padding: 0px 10px; + } + + a { + color: @gray-dark; + display: block; + line-height: 32px; + padding: 0px 10px; + text-decoration: none; + + &:hover { + background: @gray-light; + } + } + + button { + align-items: center; + background: @white; + border: 0px; + cursor: pointer; + display: flex; + margin: 0px; + padding: 0px 10px; + width: 100%; + + &:hover { + background: @gray-light; + } + + &[disabled] { + color: @gray; + cursor: wait; + + &:hover { + background: @gray-light; + } + + svg { + fill: @gray; + } + } + + span { + flex: 1; + line-height: 32px; + padding-left: 10px; + text-align: left; + } + + svg { + display: inline-block; + height: 22px; + width: 22px; + } + } +} + +.drawer > section:first-of-type ul { + border-top: 0px; +} + +:global { + .fade-enter { + opacity: 0.01; + + &.fade-enter-active { + opacity: 1; + transition: opacity 150ms ease-in; + } + } + + .fade-leave { + opacity: 1; + + &.fade-leave-active { + opacity: 0.01; + transition: opacity 150ms ease-in; + } + } +} diff --git a/web/src/shared/components/duration.js b/web/src/shared/components/duration.js new file mode 100644 index 000000000..07a875af2 --- /dev/null +++ b/web/src/shared/components/duration.js @@ -0,0 +1,10 @@ +import humanizeDuration from "humanize-duration"; +import React from "react"; + +export default class Duration extends React.Component { + render() { + const { start, finished } = this.props; + + return <time>{humanizeDuration((finished - start) * 1000)}</time>; + } +} diff --git a/web/src/shared/components/icons/back.js b/web/src/shared/components/icons/back.js new file mode 100644 index 000000000..d2bb30a52 --- /dev/null +++ b/web/src/shared/components/icons/back.js @@ -0,0 +1,17 @@ +import React, { Component } from "react"; + +export default class BackIcon extends Component { + render() { + return ( + <svg + className={this.props.className} + width={this.props.size || 24} + height={this.props.size || 24} + viewBox="0 0 24 24" + > + <path d="M0 0h24v24H0z" fill="none" /> + <path d="M20 11H7.83l5.59-5.59L12 4l-8 8 8 8 1.41-1.41L7.83 13H20v-2z" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/branch.js b/web/src/shared/components/icons/branch.js new file mode 100644 index 000000000..09b5e4de2 --- /dev/null +++ b/web/src/shared/components/icons/branch.js @@ -0,0 +1,11 @@ +import React, { Component } from "react"; + +export default class BranchIcon extends Component { + render() { + return ( + <svg viewBox="0 0 24 24"> + <path d="M6,2A3,3 0 0,1 9,5C9,6.28 8.19,7.38 7.06,7.81C7.15,8.27 7.39,8.83 8,9.63C9,10.92 11,12.83 12,14.17C13,12.83 15,10.92 16,9.63C16.61,8.83 16.85,8.27 16.94,7.81C15.81,7.38 15,6.28 15,5A3,3 0 0,1 18,2A3,3 0 0,1 21,5C21,6.32 20.14,7.45 18.95,7.85C18.87,8.37 18.64,9 18,9.83C17,11.17 15,13.08 14,14.38C13.39,15.17 13.15,15.73 13.06,16.19C14.19,16.62 15,17.72 15,19A3,3 0 0,1 12,22A3,3 0 0,1 9,19C9,17.72 9.81,16.62 10.94,16.19C10.85,15.73 10.61,15.17 10,14.38C9,13.08 7,11.17 6,9.83C5.36,9 5.13,8.37 5.05,7.85C3.86,7.45 3,6.32 3,5A3,3 0 0,1 6,2M6,4A1,1 0 0,0 5,5A1,1 0 0,0 6,6A1,1 0 0,0 7,5A1,1 0 0,0 6,4M18,4A1,1 0 0,0 17,5A1,1 0 0,0 18,6A1,1 0 0,0 19,5A1,1 0 0,0 18,4M12,18A1,1 0 0,0 11,19A1,1 0 0,0 12,20A1,1 0 0,0 13,19A1,1 0 0,0 12,18Z" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/check.js b/web/src/shared/components/icons/check.js new file mode 100644 index 000000000..8eed935b1 --- /dev/null +++ b/web/src/shared/components/icons/check.js @@ -0,0 +1,17 @@ +import React, { Component } from "react"; + +export default class CheckIcon extends Component { + render() { + return ( + <svg + className={this.props.className} + width={this.props.size || 24} + height={this.props.size || 24} + viewBox="0 0 24 24" + > + <path d="M0 0h24v24H0z" fill="none" /> + <path d="M9 16.2L4.8 12l-1.4 1.4L9 19 21 7l-1.4-1.4L9 16.2z" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/clock.js b/web/src/shared/components/icons/clock.js new file mode 100644 index 000000000..1e971cb62 --- /dev/null +++ b/web/src/shared/components/icons/clock.js @@ -0,0 +1,17 @@ +import React, { Component } from "react"; + +export default class ClockIcon extends Component { + render() { + return ( + <svg + className={this.props.className} + width={this.props.size || 24} + height={this.props.size || 24} + viewBox="0 0 24 24" + > + <path d="M0 0h24v24H0z" fill="none" /> + <path d="M22 5.72l-4.6-3.86-1.29 1.53 4.6 3.86L22 5.72zM7.88 3.39L6.6 1.86 2 5.71l1.29 1.53 4.59-3.85zM12.5 8H11v6l4.75 2.85.75-1.23-4-2.37V8zM12 4c-4.97 0-9 4.03-9 9s4.02 9 9 9c4.97 0 9-4.03 9-9s-4.03-9-9-9zm0 16c-3.87 0-7-3.13-7-7s3.13-7 7-7 7 3.13 7 7-3.13 7-7 7z" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/close.js b/web/src/shared/components/icons/close.js new file mode 100644 index 000000000..18c985726 --- /dev/null +++ b/web/src/shared/components/icons/close.js @@ -0,0 +1,17 @@ +import React, { Component } from "react"; + +export default class CloseIcon extends Component { + render() { + return ( + <svg + className={this.props.className} + width={this.props.size || 24} + height={this.props.size || 24} + viewBox="0 0 24 24" + > + <path d="M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z" /> + <path d="M0 0h24v24H0z" fill="none" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/commit.js b/web/src/shared/components/icons/commit.js new file mode 100644 index 000000000..3dbbfe20b --- /dev/null +++ b/web/src/shared/components/icons/commit.js @@ -0,0 +1,16 @@ +import React, { Component } from "react"; + +export default class CommitIcon extends Component { + render() { + return ( + <svg + className={this.props.className} + width={this.props.size || 24} + height={this.props.size || 24} + viewBox="0 0 24 24" + > + <path d="M17,12C17,14.42 15.28,16.44 13,16.9V21H11V16.9C8.72,16.44 7,14.42 7,12C7,9.58 8.72,7.56 11,7.1V3H13V7.1C15.28,7.56 17,9.58 17,12M12,9A3,3 0 0,0 9,12A3,3 0 0,0 12,15A3,3 0 0,0 15,12A3,3 0 0,0 12,9Z" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/deploy.js b/web/src/shared/components/icons/deploy.js new file mode 100644 index 000000000..273803157 --- /dev/null +++ b/web/src/shared/components/icons/deploy.js @@ -0,0 +1,11 @@ +import React, { Component } from "react"; + +export default class DeployIcon extends Component { + render() { + return ( + <svg className={this.props.className} viewBox="0 0 24 24"> + <path d="M19,18H6A4,4 0 0,1 2,14A4,4 0 0,1 6,10H6.71C7.37,7.69 9.5,6 12,6A5.5,5.5 0 0,1 17.5,11.5V12H19A3,3 0 0,1 22,15A3,3 0 0,1 19,18M19.35,10.03C18.67,6.59 15.64,4 12,4C9.11,4 6.6,5.64 5.35,8.03C2.34,8.36 0,10.9 0,14A6,6 0 0,0 6,20H19A5,5 0 0,0 24,15C24,12.36 21.95,10.22 19.35,10.03Z" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/expand.js b/web/src/shared/components/icons/expand.js new file mode 100644 index 000000000..87a9c05f6 --- /dev/null +++ b/web/src/shared/components/icons/expand.js @@ -0,0 +1,17 @@ +import React, { Component } from "react"; + +export default class ExpandIcon extends Component { + render() { + return ( + <svg + className={this.props.className} + width={this.props.size || 24} + height={this.props.size || 24} + viewBox="0 0 24 24" + > + <path d="M7.41 7.84L12 12.42l4.59-4.58L18 9.25l-6 6-6-6z" /> + <path d="M0-.75h24v24H0z" fill="none" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/index.js b/web/src/shared/components/icons/index.js new file mode 100644 index 000000000..bc5de0359 --- /dev/null +++ b/web/src/shared/components/icons/index.js @@ -0,0 +1,45 @@ +import BackIcon from "./back"; +import BranchIcon from "./branch"; +import CheckIcon from "./check"; +import ClockIcon from "./clock"; +import CloseIcon from "./close"; +import CommitIcon from "./commit"; +import DeployIcon from "./deploy"; +import ExpandIcon from "./expand"; +import LaunchIcon from "./launch"; +import LinkIcon from "./link"; +import MenuIcon from "./menu"; +import MergeIcon from "./merge"; +import PauseIcon from "./pause"; +import PlayIcon from "./play"; +import RefreshIcon from "./refresh"; +import RemoveIcon from "./remove"; +import ScheduleIcon from "./schedule"; +import StarIcon from "./star"; +import SyncIcon from "./sync"; +import TagIcon from "./tag"; +import TimelapseIcon from "./timelapse"; + +export { + BackIcon, + BranchIcon, + CheckIcon, + CloseIcon, + ClockIcon, + CommitIcon, + DeployIcon, + ExpandIcon, + LaunchIcon, + LinkIcon, + MenuIcon, + MergeIcon, + PauseIcon, + PlayIcon, + RefreshIcon, + RemoveIcon, + ScheduleIcon, + StarIcon, + SyncIcon, + TagIcon, + TimelapseIcon, +}; diff --git a/web/src/shared/components/icons/launch.js b/web/src/shared/components/icons/launch.js new file mode 100644 index 000000000..b9633fd88 --- /dev/null +++ b/web/src/shared/components/icons/launch.js @@ -0,0 +1,12 @@ +import React, { Component } from "react"; + +export default class LaunchIcon extends Component { + render() { + return ( + <svg className={this.props.className} viewBox="0 0 24 24"> + <path d="M0 0h24v24H0z" fill="none" /> + <path d="M19 19H5V5h7V3H5c-1.11 0-2 .9-2 2v14c0 1.1.89 2 2 2h14c1.1 0 2-.9 2-2v-7h-2v7zM14 3v2h3.59l-9.83 9.83 1.41 1.41L19 6.41V10h2V3h-7z" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/link.js b/web/src/shared/components/icons/link.js new file mode 100644 index 000000000..749eec60f --- /dev/null +++ b/web/src/shared/components/icons/link.js @@ -0,0 +1,17 @@ +import React, { Component } from "react"; + +export default class LinkIcon extends Component { + render() { + return ( + <svg + className={this.props.className} + width={this.props.size || 24} + height={this.props.size || 24} + viewBox="0 0 24 24" + > + <path d="M0 0h24v24H0z" fill="none" /> + <path d="M3.9 12c0-1.71 1.39-3.1 3.1-3.1h4V7H7c-2.76 0-5 2.24-5 5s2.24 5 5 5h4v-1.9H7c-1.71 0-3.1-1.39-3.1-3.1zM8 13h8v-2H8v2zm9-6h-4v1.9h4c1.71 0 3.1 1.39 3.1 3.1s-1.39 3.1-3.1 3.1h-4V17h4c2.76 0 5-2.24 5-5s-2.24-5-5-5z" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/menu.js b/web/src/shared/components/icons/menu.js new file mode 100644 index 000000000..24aa4752e --- /dev/null +++ b/web/src/shared/components/icons/menu.js @@ -0,0 +1,17 @@ +import React, { Component } from "react"; + +export default class MenuIcon extends Component { + render() { + return ( + <svg + className={this.props.className} + width={this.props.size || 24} + height={this.props.size || 24} + viewBox="0 0 24 24" + > + <path d="M0 0h24v24H0z" fill="none" /> + <path d="M3 18h18v-2H3v2zm0-5h18v-2H3v2zm0-7v2h18V6H3z" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/merge.js b/web/src/shared/components/icons/merge.js new file mode 100644 index 000000000..a45e5b3c3 --- /dev/null +++ b/web/src/shared/components/icons/merge.js @@ -0,0 +1,16 @@ +import React, { Component } from "react"; + +export default class MergeIcon extends Component { + render() { + return ( + <svg className={this.props.className} viewBox="0 0 24 24"> + <path d="M5.41,21L6.12,17H2.12L2.47,15H6.47L7.53,9H3.53L3.88,7H7.88L8.59,3H10.59L9.88,7H15.88L16.59,3H18.59L17.88,7H21.88L21.53,9H17.53L16.47,15H20.47L20.12,17H16.12L15.41,21H13.41L14.12,17H8.12L7.41,21H5.41M9.53,9L8.47,15H14.47L15.53,9H9.53Z" /> + </svg> + ); + } +} + +// <svg class={this.props.className} viewBox="0 0 54.5 68"> +// <path d="M20,13C20,8.6,16.4,5,12.1,5C7.7,5,4.2,8.6,4.2,13c0,3.2,1.9,6,4.7,7.2v27.1c-2.7,1.2-4.7,4-4.7,7.2c0,4.4,3.6,7.9,7.9,7.9 c4.4,0,7.9-3.6,7.9-7.9c0-3.2-1.9-6-4.7-7.2V20.2C18.1,18.9,20,16.2,20,13z M16,54.5c0,2.2-1.8,3.9-3.9,3.9c-2.2,0-3.9-1.8-3.9-3.9 c0-2.2,1.8-3.9,3.9-3.9C14.2,50.5,16,52.3,16,54.5z M12.1,16.9c-2.2,0-3.9-1.8-3.9-3.9c0-2.2,1.8-3.9,3.9-3.9C14.2,9,16,10.8,16,13 C16,15.1,14.2,16.9,12.1,16.9z"/> +// <path d="M45.3,47.3V20.8c0-6.1-5-11.1-11.1-11.1h-2.7V3.6L20.7,13l10.8,9.3v-6.1h2.7c2.6,0,4.6,2.1,4.6,4.6v26.4 c-2.7,1.2-4.7,4-4.7,7.2c0,4.4,3.6,7.9,7.9,7.9c4.4,0,7.9-3.6,7.9-7.9C50,51.3,48.1,48.5,45.3,47.3z M42.1,58.4 c-2.2,0-3.9-1.8-3.9-3.9c0-2.2,1.8-3.9,3.9-3.9c2.2,0,3.9,1.8,3.9,3.9C46,56.6,44.2,58.4,42.1,58.4z"/> +// </svg> diff --git a/web/src/shared/components/icons/pause.js b/web/src/shared/components/icons/pause.js new file mode 100644 index 000000000..f7378fbb7 --- /dev/null +++ b/web/src/shared/components/icons/pause.js @@ -0,0 +1,17 @@ +import React, { Component } from "react"; + +export default class PauseIcon extends Component { + render() { + return ( + <svg + className={this.props.className} + width={this.props.size || 24} + height={this.props.size || 24} + viewBox="0 0 24 24" + > + <path d="M6 19h4V5H6v14zm8-14v14h4V5h-4z" /> + <path d="M0 0h24v24H0z" fill="none" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/play.js b/web/src/shared/components/icons/play.js new file mode 100644 index 000000000..b12697a1c --- /dev/null +++ b/web/src/shared/components/icons/play.js @@ -0,0 +1,17 @@ +import React, { Component } from "react"; + +export default class PlayIcon extends Component { + render() { + return ( + <svg + className={this.props.className} + width={this.props.size || 24} + height={this.props.size || 24} + viewBox="0 0 24 24" + > + <path d="M8 5v14l11-7z" /> + <path d="M0 0h24v24H0z" fill="none" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/refresh.js b/web/src/shared/components/icons/refresh.js new file mode 100644 index 000000000..659dd8692 --- /dev/null +++ b/web/src/shared/components/icons/refresh.js @@ -0,0 +1,17 @@ +import React, { Component } from "react"; + +export default class RefreshIcon extends Component { + render() { + return ( + <svg + className={this.props.className} + width={this.props.size || 24} + height={this.props.size || 24} + viewBox="0 0 24 24" + > + <path d="M17.65 6.35C16.2 4.9 14.21 4 12 4c-4.42 0-7.99 3.58-7.99 8s3.57 8 7.99 8c3.73 0 6.84-2.55 7.73-6h-2.08c-.82 2.33-3.04 4-5.65 4-3.31 0-6-2.69-6-6s2.69-6 6-6c1.66 0 3.14.69 4.22 1.78L13 11h7V4l-2.35 2.35z" /> + <path d="M0 0h24v24H0z" fill="none" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/remove.js b/web/src/shared/components/icons/remove.js new file mode 100644 index 000000000..0a791b21e --- /dev/null +++ b/web/src/shared/components/icons/remove.js @@ -0,0 +1,17 @@ +import React, { Component } from "react"; + +export default class CheckIcon extends Component { + render() { + return ( + <svg + className={this.props.className} + width={this.props.size || 24} + height={this.props.size || 24} + viewBox="0 0 24 24" + > + <path d="M19 13H5v-2h14v2z" /> + <path d="M0 0h24v24H0z" fill="none" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/report.js b/web/src/shared/components/icons/report.js new file mode 100644 index 000000000..5a87303a0 --- /dev/null +++ b/web/src/shared/components/icons/report.js @@ -0,0 +1,12 @@ +import React, { Component } from "react"; + +export default class ReportIcon extends Component { + render() { + return ( + <svg className={this.props.className} viewBox="0 0 24 24"> + <path d="M15.73 3H8.27L3 8.27v7.46L8.27 21h7.46L21 15.73V8.27L15.73 3zM12 17.3c-.72 0-1.3-.58-1.3-1.3 0-.72.58-1.3 1.3-1.3.72 0 1.3.58 1.3 1.3 0 .72-.58 1.3-1.3 1.3zm1-4.3h-2V7h2v6z" /> + <path d="M0 0h24v24H0z" fill="none" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/schedule.js b/web/src/shared/components/icons/schedule.js new file mode 100644 index 000000000..ad5ad2693 --- /dev/null +++ b/web/src/shared/components/icons/schedule.js @@ -0,0 +1,18 @@ +import React, { Component } from "react"; + +export default class ScheduleIcon extends Component { + render() { + return ( + <svg + className={this.props.className} + width={this.props.size || 24} + height={this.props.size || 24} + viewBox="0 0 24 24" + > + <path d="M11.99 2C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z" /> + <path d="M0 0h24v24H0z" fill="none" /> + <path d="M12.5 7H11v6l5.25 3.15.75-1.23-4.5-2.67z" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/star.js b/web/src/shared/components/icons/star.js new file mode 100644 index 000000000..c9e394255 --- /dev/null +++ b/web/src/shared/components/icons/star.js @@ -0,0 +1,20 @@ +import React, { Component } from "react"; + +export default class StarIcon extends Component { + render() { + return ( + <svg + className={this.props.className} + width={this.props.size || 24} + height={this.props.size || 24} + viewBox="0 0 512 512" + > + {this.props.filled === true ? ( + <path d="M256 372.686L380.83 448l-33.021-142.066L458 210.409l-145.267-12.475L256 64l-56.743 133.934L54 210.409l110.192 95.525L131.161 448z" /> + ) : ( + <path d="M458 210.409l-145.267-12.476L256 64l-56.743 133.934L54 210.409l110.192 95.524L131.161 448 256 372.686 380.83 448l-33.021-142.066L458 210.409zM272.531 345.286L256 335.312l-16.53 9.973-59.988 36.191 15.879-68.296 4.369-18.79-14.577-12.637-52.994-45.939 69.836-5.998 19.206-1.65 7.521-17.75 27.276-64.381 27.27 64.379 7.52 17.751 19.208 1.65 69.846 5.998-52.993 45.939-14.576 12.636 4.367 18.788 15.875 68.299-59.984-36.189z" /> + )} + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/sync.js b/web/src/shared/components/icons/sync.js new file mode 100644 index 000000000..e102bfa4a --- /dev/null +++ b/web/src/shared/components/icons/sync.js @@ -0,0 +1,12 @@ +import React, { Component } from "react"; + +export default class SyncIcon extends Component { + render() { + return ( + <svg className={this.props.className} viewBox="0 0 24 24"> + <path d="M12 4V1L8 5l4 4V6c3.31 0 6 2.69 6 6 0 1.01-.25 1.97-.7 2.8l1.46 1.46C19.54 15.03 20 13.57 20 12c0-4.42-3.58-8-8-8zm0 14c-3.31 0-6-2.69-6-6 0-1.01.25-1.97.7-2.8L5.24 7.74C4.46 8.97 4 10.43 4 12c0 4.42 3.58 8 8 8v3l4-4-4-4v3z" /> + <path d="M0 0h24v24H0z" fill="none" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/tag.js b/web/src/shared/components/icons/tag.js new file mode 100644 index 000000000..eb68cc1be --- /dev/null +++ b/web/src/shared/components/icons/tag.js @@ -0,0 +1,11 @@ +import React, { Component } from "react"; + +export default class TagIcon extends Component { + render() { + return ( + <svg className={this.props.className} viewBox="0 0 24 24"> + <path d="M5.5,7A1.5,1.5 0 0,0 7,5.5A1.5,1.5 0 0,0 5.5,4A1.5,1.5 0 0,0 4,5.5A1.5,1.5 0 0,0 5.5,7M21.41,11.58C21.77,11.94 22,12.44 22,13C22,13.55 21.78,14.05 21.41,14.41L14.41,21.41C14.05,21.77 13.55,22 13,22C12.45,22 11.95,21.77 11.58,21.41L2.59,12.41C2.22,12.05 2,11.55 2,11V4C2,2.89 2.89,2 4,2H11C11.55,2 12.05,2.22 12.41,2.58L21.41,11.58M13,20L20,13L11.5,4.5L4.5,11.5L13,20Z" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/icons/timelapse.js b/web/src/shared/components/icons/timelapse.js new file mode 100644 index 000000000..e67239fb8 --- /dev/null +++ b/web/src/shared/components/icons/timelapse.js @@ -0,0 +1,17 @@ +import React, { Component } from "react"; + +export default class TimelapseIcon extends Component { + render() { + return ( + <svg + className={this.props.className} + width={this.props.size || 24} + height={this.props.size || 24} + viewBox="0 0 24 24" + > + <path d="M0 0h24v24H0z" fill="none" /> + <path d="M16.24 7.76C15.07 6.59 13.54 6 12 6v6l-4.24 4.24c2.34 2.34 6.14 2.34 8.49 0 2.34-2.34 2.34-6.14-.01-8.48zM12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 18c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z" /> + </svg> + ); + } +} diff --git a/web/src/shared/components/logo.js b/web/src/shared/components/logo.js new file mode 100644 index 000000000..bdfad017b --- /dev/null +++ b/web/src/shared/components/logo.js @@ -0,0 +1,17 @@ +import React, { Component } from "react"; + +export default class Logo extends Component { + render() { + return ( + <svg viewBox="0 0 50 62.5" preserveAspectRatio="xMidYMid"> + <g> + <path + fillRule="evenodd" + clipRule="evenodd" + d="M15.872,0.468c1.148,1.088,1.582,2.188,2.855,2.337l0.036,0.007c-0.588,0.606-1.089,1.402-1.443,2.423c-0.379,1.096-0.488,2.285-0.614,3.659c-0.189,2.046-0.401,4.364-1.556,7.269 c-2.486,6.258-1.119,11.631,0.332,17.317c0.664,2.604,1.348,5.297,1.642,8.107c0.035,0.355,0.287,0.652,0.633,0.744 c0.346,0.095,0.709-0.035,0.922-0.323c0.227-0.313,0.524-0.797,0.86-1.424c0.84,3.323,1.355,6.131,1.783,8.697 c0.126,0.73,1.048,0.973,1.517,0.41c2.881-3.463,3.763-8.636,2.184-12.674c0.459-2.433,1.402-4.45,2.398-6.583 c0.536-1.15,1.08-2.318,1.55-3.566c0.228-0.084,0.569-0.314,0.791-0.441l1.706-0.981l-0.256,1.052 c-0.112,0.461,0.171,0.929,0.635,1.04c0.457,0.118,0.93-0.173,1.043-0.632l0.68-2.858l1.285-2.95 c0.19-0.436-0.009-0.943-0.446-1.135c-0.44-0.189-0.947,0.01-1.135,0.448l-1.152,2.669l-2.383,1.372 c0.235-0.932,0.414-1.919,0.508-2.981c0.432-4.859-0.718-9.074-3.066-11.266c-0.163-0.157-0.208-0.281-0.247-0.26 c0.095-0.119,0.249-0.26,0.358-0.374c2.283-1.693,6.047-0.147,8.319,0.751c0.589,0.231,0.876-0.338,0.316-0.67 c-1.949-1.154-5.948-4.197-8.188-6.194c-0.313-0.275-0.527-0.607-0.89-0.913c-2.415-4.266-8.168-1.764-10.885-2.252 C15.862,0.275,15.798,0.396,15.872,0.468 M26.852,6.367c-0.059,1.242-0.603,1.8-0.999,2.208c-0.218,0.224-0.427,0.436-0.525,0.738 c-0.236,0.714,0.008,1.51,0.66,2.143c1.974,1.84,2.925,5.527,2.538,9.861c-0.291,3.287-1.448,5.762-2.671,8.384 c-1.031,2.207-2.096,4.489-2.577,7.259c-0.027,0.161-0.01,0.33,0.056,0.481c1.021,2.433,1.135,6.196-0.672,9.46 c-0.461-2.553-1.053-5.385-1.97-8.712c1.964-4.488,4.203-11.75,2.919-17.668c-0.325-1.497-1.304-3.276-2.387-4.207 c-0.208-0.179-0.402-0.237-0.495-0.167c-0.084,0.061-0.151,0.238-0.062,0.444c0.55,1.266,0.879,2.599,1.226,4.276 c1.125,5.443-0.956,12.49-2.835,16.782l-0.116,0.259l-0.457,0.982c-0.356-2.014-0.849-3.95-1.33-5.839 c-1.379-5.407-2.679-10.516-0.401-16.255c1.247-3.137,1.483-5.692,1.672-7.746c0.116-1.263,0.216-2.355,0.526-3.252 c0.905-2.605,3.062-3.178,4.744-2.852C25.328,3.262,26.936,4.539,26.852,6.367z M23.984,6.988c0.617,0.204,1.283-0.131,1.487-0.75 c0.202-0.617-0.134-1.283-0.751-1.487c-0.618-0.204-1.285,0.134-1.487,0.751C23.029,6.12,23.366,6.786,23.984,6.988z" + /> + </g> + </svg> + ); + } +} diff --git a/web/src/shared/components/menu.js b/web/src/shared/components/menu.js new file mode 100644 index 000000000..389d1e54b --- /dev/null +++ b/web/src/shared/components/menu.js @@ -0,0 +1,32 @@ +import React, { Component } from "react"; +import { NavLink as Link } from "react-router-dom"; +import PropTypes from "prop-types"; + +import styles from "./menu.less"; + +export default class Menu extends Component { + propTypes = { items: PropTypes.array, right: PropTypes.any }; + render() { + const items = this.props.items; + const right = this.props.right ? ( + <div className={styles.right}>{this.props.right}</div> + ) : null; + return ( + <section className={styles.root}> + <div className={styles.left}> + {items.map(i => ( + <Link + key={i.to + i.label} + to={i.to} + exact={true} + activeClassName={styles["link-active"]} + > + {i.label} + </Link> + ))} + </div> + {right} + </section> + ); + } +} diff --git a/web/src/shared/components/menu.less b/web/src/shared/components/menu.less new file mode 100644 index 000000000..1afeb9989 --- /dev/null +++ b/web/src/shared/components/menu.less @@ -0,0 +1,65 @@ +@import '~shared/styles/colors'; + +@itemsHeight: 32px; + +.left { + flex: 1; +} + +.right { + button { + border: 1px solid @gray-light; + font-size: 12px; + height: @itemsHeight; + outline: none; + + &:hover { + border-color: @green; + color: @green; + cursor: pointer; + + svg { + fill: @green; + } + } + + svg, span { + display: inline-block; + vertical-align: middle; + } + + svg { + width: 24px; + height: 24px; + } + + span { + font-size: 14px; + } + } +} + + +.root { + padding: 20px; + display: flex; + flex-direction: row; + border-bottom: 1px solid @gray-light; + + a{ + display: inline-block; + vertical-align: top; + color: #000; + text-decoration: none; + padding: 0 12px; + height: @itemsHeight; + line-height: @itemsHeight; + margin-right: 12px; + border-bottom: 2px solid transparent; + } + +} + +a.link-active { + border-bottom-color: @green; +} \ No newline at end of file diff --git a/web/src/shared/components/snackbar.js b/web/src/shared/components/snackbar.js new file mode 100644 index 000000000..db76c4fd8 --- /dev/null +++ b/web/src/shared/components/snackbar.js @@ -0,0 +1,47 @@ +import React from "react"; +import styles from "./snackbar.less"; +import CloseIcon from "shared/components/icons/close"; +import { CSSTransitionGroup } from "react-transition-group"; + +export class Snackbar extends React.Component { + render() { + const { message } = this.props; + + let classes = [styles.snackbar]; + if (message) { + classes.push(styles.open); + } + + const content = message ? ( + <div className={classes.join(" ")} key={message}> + <div>{message}</div> + <button onClick={this.props.onClose}> + <CloseIcon /> + </button> + </div> + ) : null; + + return ( + <CSSTransitionGroup + transitionName="slideup" + transitionEnterTimeout={200} + transitionLeaveTimeout={200} + transitionAppearTimeout={200} + transitionAppear={true} + transitionEnter={true} + transitionLeave={true} + className={classes.root} + > + {content} + </CSSTransitionGroup> + ); + } +} + +// const SnackbarContent = ({ children, ...props }) => { +// <div {...props}>{children}</div> +// } +// +// const SnackbarClose = ({ children, ...props }) => { +// <div {...props}>{children}</div> +// } diff --git a/web/src/shared/components/snackbar.less b/web/src/shared/components/snackbar.less new file mode 100644 index 000000000..cc88a90bb --- /dev/null +++ b/web/src/shared/components/snackbar.less @@ -0,0 +1,78 @@ +@import '~shared/styles/colors'; + +@snackbar-margin: 20px; +@snackbar-min-width: 500px; + +// wrapper for the css transition group +.root { + bottom: -1000px; + height: 0px; + position: absolute; + top: -1000px; + width: 0px; +} + +.snackbar { + align-items: stretch; + background: @gray-dark; + bottom: @snackbar-margin; + box-shadow: rgba(0, 0, 0, 0.2), 0px 6px 10px 0px rgba(0, 0, 0, 0.14), 0px 1px + 18px 0px rgba(0, 0, 0, 0.12); + display: none; + flex-direction: row; + left: @snackbar-margin; + min-width: @snackbar-min-width; + position: fixed; + z-index: 2; + + &.open { + display: flex; + } + + &> :first-child { + color: @white; + flex: 1; + font-size: 14px; + line-height: 24px; + padding: 10px 20px; + vertical-align: middle; + } + + button { + background: transparent; + border: 0px; + cursor: pointer; + display: flex; + flex: 0 0 24px; + margin: 0px; + margin-right: 10px; + outline: none; + padding: 0px; + + svg { + align-items: center; + fill: @white; + height: 24px; + } + } +} + +:global { + .slideup-enter { + bottom: -50px; + } + + .slideup-enter.slideup-enter-active { + bottom: @snackbar-margin; + transition: bottom 200ms linear; + } + + .slideup-leave { + bottom: @snackbar-margin; + } + + .slideup-leave.slideup-leave-active { + bottom: -50px; + transition: bottom 200ms linear; + } +} diff --git a/web/src/shared/components/status.js b/web/src/shared/components/status.js new file mode 100644 index 000000000..18a17a16b --- /dev/null +++ b/web/src/shared/components/status.js @@ -0,0 +1,100 @@ +import React, { Component } from "react"; +import classnames from "classnames"; +import { + STATUS_BLOCKED, + STATUS_DECLINED, + STATUS_ERROR, + STATUS_FAILURE, + STATUS_KILLED, + STATUS_PENDING, + STATUS_RUNNING, + STATUS_SKIPPED, + STATUS_STARTED, + STATUS_SUCCESS, +} from "shared/constants/status"; +import style from "./status.less"; + +import { + CheckIcon, + CloseIcon, + ClockIcon, + RefreshIcon, + RemoveIcon, +} from "./icons/index"; + +const defaultIconSize = 15; + +const statusLabel = status => { + switch (status) { + case STATUS_BLOCKED: + return "Pending Approval"; + case STATUS_DECLINED: + return "Declined"; + case STATUS_ERROR: + return "Error"; + case STATUS_FAILURE: + return "Failure"; + case STATUS_KILLED: + return "Cancelled"; + case STATUS_PENDING: + return "Pending"; + case STATUS_RUNNING: + return "Running"; + case STATUS_SKIPPED: + return "Skipped"; + case STATUS_STARTED: + return "Running"; + case STATUS_SUCCESS: + return "Successful"; + default: + return ""; + } +}; + +const renderIcon = (status, size) => { + switch (status) { + case STATUS_SKIPPED: + return <RemoveIcon size={size} />; + case STATUS_PENDING: + return <ClockIcon size={size} />; + case STATUS_RUNNING: + case STATUS_STARTED: + return <RefreshIcon size={size} />; + case STATUS_SUCCESS: + return <CheckIcon size={size} />; + default: + return <CloseIcon size={size} />; + } +}; + +export default class Status extends Component { + shouldComponentUpdate(nextProps, nextState) { + return this.props.status !== nextProps.status; + } + + render() { + const { status } = this.props; + const icon = renderIcon(status, defaultIconSize); + const classes = classnames(style.root, style[status]); + return <div className={classes}>{icon}</div>; + } +} + +export const StatusLabel = ({ status }) => { + return ( + <div className={classnames(style.label, style[status])}> + <div>{statusLabel(status)}</div> + </div> + ); +}; + +export const StatusText = ({ status, text }) => { + return ( + <div + className={classnames(style.label, style[status])} + style="text-transform: capitalize;padding: 5px 10px;" + > + <div>{text}</div> + </div> + ); +}; diff --git a/web/src/shared/components/status.less b/web/src/shared/components/status.less new file mode 100644 index 000000000..f6e553a7e --- /dev/null +++ b/web/src/shared/components/status.less @@ -0,0 +1,204 @@ +@import '~shared/styles/colors'; + +.root { + align-content: center; + border-radius: 50%; + border-style: solid; + border-width: 2px; + box-sizing: border-box; + display: flex; + height: 23px; + padding: 2px; + width: 23px; + + // + // success + // + + &.success { + border-color: @green; + + svg { + fill: @green; + } + } + + // + // failure + // + + &.declined, + &.failure, + &.killed, + &.error { + border-color: @red; + + svg { + fill: @red; + } + } + + // + // blocked, running, started + // + + &.blocked, + &.running, + &.started { + border-color: @yellow; + + svg { + fill: @yellow; + } + } + + &.started svg, + &.running svg { + animation: spinner 1.2s ease infinite; + } + + // + // pending, skipped + // + + &.pending, + &.skipped { + border-color: @gray; + + svg { + fill: @gray; + } + } + + &.pending svg { + animation: wrench 2.5s ease infinite; + transform-origin: center 54%; + } +} + +@keyframes spinner { + 0% { + transform: rotate(0); + } + + 100% { + transform: rotate(359deg); + } +} + +@keyframes wrench { + 0% { + transform: rotate(-12deg); + } + + 8% { + transform: rotate(12deg); + } + + 10% { + transform: rotate(24deg); + } + + 18% { + transform: rotate(-24deg); + } + + 20% { + transform: rotate(-24deg); + } + + 28% { + transform: rotate(24deg); + } + + 30% { + transform: rotate(24deg); + } + + 38% { + transform: rotate(-24deg); + } + + 40% { + transform: rotate(-24deg); + } + + 48% { + transform: rotate(24deg); + } + + 50% { + transform: rotate(24deg); + } + + 58% { + transform: rotate(-24deg); + } + + 60% { + transform: rotate(-24deg); + } + + 68% { + transform: rotate(24deg); + } + + 75%, + 100% { + transform: rotate(0); + } +} + +.label { + background-color: @green; + border-radius: 2px; + color: @white; + display: flex; + padding: 10px 20px; + text-shadow: 0px 1px 2px rgba(0, 0, 0, 0.1); + + div { + flex: 1; + font-size: 15px; + line-height: 22px; + vertical-align: middle; + } + + // + // success + // + + &.success { + background-color: @green; + } + + // + // failure + // + + &.declined, + &.failure, + &.killed, + &.error { + background-color: @red; + } + + // + // blocked, running, started + // + + &.blocked, + &.running, + &.started { + background-color: @yellow; + } + + // + // pending, skipped + // + + &.pending, + &.skipped { + background-color: @gray; + } +} diff --git a/web/src/shared/components/status_number.js b/web/src/shared/components/status_number.js new file mode 100644 index 000000000..f030605e9 --- /dev/null +++ b/web/src/shared/components/status_number.js @@ -0,0 +1,12 @@ +import React, { Component } from "react"; +import classnames from "classnames"; + +import styles from "./status_number.less"; + +export default class StatusNumber extends Component { + render() { + const { status, number } = this.props; + const className = classnames(styles.root, styles[status]); + return <div className={className}>{number}</div>; + } +} diff --git a/web/src/shared/components/status_number.less b/web/src/shared/components/status_number.less new file mode 100644 index 000000000..f5fcc575f --- /dev/null +++ b/web/src/shared/components/status_number.less @@ -0,0 +1,54 @@ +@import '~shared/styles/colors'; + +.root { + border-radius: 2px; + border-style: solid; + border-width: 2px; + display: inline-block; + font-size: 14px; + line-height: 20px; + min-width: 65px; + text-align: center; + + // + // success + // + + &.success { + border-color: @green; + color: @green; + } + + // + // failure + // + + &.declined, + &.failure, + &.killed, + &.error { + border-color: @red; + color: @red; + } + + // + // blocked, running, started + // + + &.blocked, + &.running, + &.started { + border-color: @yellow; + color: @yellow; + } + + // + // pending, skipped + // + + &.pending, + &.skipped { + border-color: @gray; + color: @gray; + } +} diff --git a/web/src/shared/components/sync.js b/web/src/shared/components/sync.js new file mode 100644 index 000000000..6f43ef021 --- /dev/null +++ b/web/src/shared/components/sync.js @@ -0,0 +1,16 @@ +import React from "react"; +import Icon from "./icons/refresh"; +import styles from "./sync.less"; + +export const Message = () => { + return ( + <div className={styles.root}> + <div className={styles.alert}> + <div> + <Icon /> + </div> + <div>Account synchronization in progress</div> + </div> + </div> + ); +}; diff --git a/web/src/shared/components/sync.less b/web/src/shared/components/sync.less new file mode 100644 index 000000000..1bd90d747 --- /dev/null +++ b/web/src/shared/components/sync.less @@ -0,0 +1,46 @@ +@import '~shared/styles/colors'; + +@font: 'Roboto'; + +.root { + box-sizing: border-box; + margin: 50px auto; + max-width: 400px; + min-width: 400px; + padding: 30px; + + .alert { + background: @yellow; + border-radius: 2px; + color: @white; + display: flex; + margin-bottom: 20px; + padding: 20px; + text-align: left; + + &> :last-child { + font-family: @font; + font-size: 15px; + line-height: 20px; + padding-left: 10px; + padding-top: 2px; + } + } + + svg { + animation: spinner 1.2s ease infinite; + fill: @white; + height: 26px; + width: 26px; + } +} + +@keyframes spinner { + 0% { + transform: rotate(0deg); + } + + 100% { + transform: rotate(359deg); + } +} diff --git a/web/src/shared/constants/events.js b/web/src/shared/constants/events.js new file mode 100644 index 000000000..a4cf86130 --- /dev/null +++ b/web/src/shared/constants/events.js @@ -0,0 +1,6 @@ +const EVENT_DEPLOY = "deployment"; +const EVENT_PULL_REQUEST = "pull_request"; +const EVENT_PUSH = "push"; +const EVENT_TAG = "tag"; + +export { EVENT_DEPLOY, EVENT_PULL_REQUEST, EVENT_PUSH, EVENT_TAG }; diff --git a/web/src/shared/constants/status.js b/web/src/shared/constants/status.js new file mode 100644 index 000000000..5be3ac197 --- /dev/null +++ b/web/src/shared/constants/status.js @@ -0,0 +1,23 @@ +const STATUS_BLOCKED = "blocked"; +const STATUS_DECLINED = "declined"; +const STATUS_ERROR = "error"; +const STATUS_FAILURE = "failure"; +const STATUS_KILLED = "killed"; +const STATUS_PENDING = "pending"; +const STATUS_RUNNING = "running"; +const STATUS_SKIPPED = "skipped"; +const STATUS_STARTED = "started"; +const STATUS_SUCCESS = "success"; + +export { + STATUS_BLOCKED, + STATUS_DECLINED, + STATUS_ERROR, + STATUS_FAILURE, + STATUS_KILLED, + STATUS_PENDING, + STATUS_RUNNING, + STATUS_SKIPPED, + STATUS_SUCCESS, + STATUS_STARTED, +}; diff --git a/web/src/shared/constants/visibility.js b/web/src/shared/constants/visibility.js new file mode 100644 index 000000000..22b79f825 --- /dev/null +++ b/web/src/shared/constants/visibility.js @@ -0,0 +1,5 @@ +const VISIBILITY_PUBLIC = "public"; +const VISIBILITY_PRIVATE = "private"; +const VISIBILITY_INTERNAL = "internal"; + +export { VISIBILITY_PUBLIC, VISIBILITY_PRIVATE, VISIBILITY_INTERNAL }; diff --git a/web/src/shared/styles/ansi.less b/web/src/shared/styles/ansi.less new file mode 100644 index 000000000..f9591e4d8 --- /dev/null +++ b/web/src/shared/styles/ansi.less @@ -0,0 +1,35 @@ +// ansi terminal color and background settings +// DO NOT EDIT +:global { + .ansi-bright-black-fg, .ansi-black-fg { color: #B3B3B3; } + .ansi-bright-red-fg, .ansi-red-fg { color: #fb9fb1; } + .ansi-bright-green-fg, .ansi-green-fg { color: #acc267; } + .ansi-bright-yellow-fg, .ansi-yellow-fg { color: #ddb26f; } + .ansi-bright-blue-fg, .ansi-blue-fg { color: #6fc2ef; } + .ansi-bright-magenta-fg, .ansi-magenta-fg { color: #e1a3ee; } + .ansi-bright-cyan-fg, .ansi-cyan-fg { color: #12cfc0; } + .ansi-bright-white-fg, .ansi-white-fg { color: #151515; } + + .ansi-bright-black-fg, .ansi-bright-red-fg, .ansi-bright-green-fg, .ansi-bright-yellow-fg, + .ansi-bright-blue-fg, .ansi-bright-magenta-fg, .ansi-bright-cyan-fg, .ansi-bright-white-fg { + font-weight: bold; + } + + .ansi-black-bg { background-color: #d0d0d0; } + .ansi-red-bg { background-color: #fb9fb1; } + .ansi-green-bg { background-color: #acc267; } + .ansi-yellow-bg { background-color: #ddb26f;} + .ansi-blue-bg { background-color: #6fc2ef; } + .ansi-magenta-bg { background-color: #e1a3ee;} + .ansi-cyan-bg { background-color: #12cfc0; } + .ansi-white-bg { background-color: #151515; } + + .ansi-bright-black-bg { background-color: #f5f5f5; } + .ansi-bright-red-bg { background-color: #fb9fb1; } + .ansi-bright-green-bg {background-color: #acc267;} + .ansi-bright-yellow-bg {background-color: #ddb26f;} + .ansi-bright-blue-bg {background-color: #6fc2ef;} + .ansi-bright-magenta-bg {background-color: #e1a3ee;} + .ansi-bright-cyan-bg {background-color: #12cfc0; } + .ansi-bright-white-bg { background-color: #505050; } +} diff --git a/web/src/shared/styles/colors.less b/web/src/shared/styles/colors.less new file mode 100644 index 000000000..fddfcdcbd --- /dev/null +++ b/web/src/shared/styles/colors.less @@ -0,0 +1,12 @@ +@white: #ffffff; +@gray-light: #eceff1; +@gray: #bdbdbd; +@gray-dark: #212121; +@black: #000000; + +@splitter-border-color: darken(@gray-light, 10%); + +// colors from https://atom.io/themes/dash-ui +@green: #4dc89a; +@red: #fc4758; +@yellow: #fdb835; diff --git a/web/src/shared/styles/utils.less b/web/src/shared/styles/utils.less new file mode 100644 index 000000000..d0ec136a8 --- /dev/null +++ b/web/src/shared/styles/utils.less @@ -0,0 +1,5 @@ +.text-ellipsis { + overflow: hidden; + text-overflow: ellipsis; + white-space: nowrap; +} diff --git a/web/src/shared/utils/build.js b/web/src/shared/utils/build.js new file mode 100644 index 000000000..0c53948aa --- /dev/null +++ b/web/src/shared/utils/build.js @@ -0,0 +1,187 @@ +import { repositorySlug } from "./repository"; +import { displayMessage } from "./message"; +import { STATUS_PENDING, STATUS_RUNNING } from "shared/constants/status"; + +/** + * Gets the build for the named repository and stores + * the results in the state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + * @param {string} owner - The repository owner. + * @param {string} name - The repository name. + * @param {number|string} number - The build number. + */ +export const fetchBuild = (tree, client, owner, name, number) => { + const slug = repositorySlug(owner, name); + + tree.unset(["builds", "loaded"]); + client + .getBuild(owner, name, number) + .then(build => { + const path = ["builds", "data", slug, build.number]; + + if (tree.exists(path)) { + tree.deepMerge(path, build); + } else { + tree.set(path, build); + } + + tree.set(["builds", "loaded"], true); + }) + .catch(error => { + tree.set(["builds", "loaded"], true); + tree.set(["builds", "error"], error); + }); +}; + +/** + * Gets the build list for the named repository and + * stores the results in the state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + * @param {string} owner - The repository owner. + * @param {string} name - The repository name. + */ +export const fetchBuildList = (tree, client, owner, name, page = 1) => { + const slug = repositorySlug(owner, name); + + tree.unset(["builds", "loaded"]); + tree.unset(["builds", "error"]); + + client + .getBuildList(owner, name, { page: page }) + .then(results => { + let list = {}; + results.map(build => { + list[build.number] = build; + }); + + const path = ["builds", "data", slug]; + if (tree.exists(path)) { + tree.deepMerge(path, list); + } else { + tree.set(path, list); + } + + tree.unset(["builds", "error"]); + tree.set(["builds", "loaded"], true); + }) + .catch(error => { + tree.set(["builds", "error"], error); + tree.set(["builds", "loaded"], true); + }); +}; + +/** + * Cancels the build. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + * @param {string} owner - The repository owner. + * @param {string} name - The repository name. + * @param {number} build - The build number. + * @param {number} proc - The process number. + */ +export const cancelBuild = (tree, client, owner, repo, build, proc) => { + client + .cancelBuild(owner, repo, build, proc) + .then(result => { + displayMessage(tree, "Successfully cancelled your build"); + }) + .catch(() => { + displayMessage(tree, "Failed to cancel your build"); + }); +}; + +/** + * Restarts the build. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + * @param {string} owner - The repository owner. + * @param {string} name - The repository name. + * @param {number} build - The build number. + */ +export const restartBuild = (tree, client, owner, repo, build) => { + client + .restartBuild(owner, repo, build, { fork: true }) + .then(result => { + displayMessage(tree, "Successfully restarted your build"); + }) + .catch(() => { + displayMessage(tree, "Failed to restart your build"); + }); +}; + +/** + * Approves the blocked build. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + * @param {string} owner - The repository owner. + * @param {string} name - The repository name. + * @param {number} build - The build number. + */ +export const approveBuild = (tree, client, owner, repo, build) => { + client + .approveBuild(owner, repo, build) + .then(result => { + displayMessage(tree, "Successfully processed your approval decision"); + }) + .catch(() => { + displayMessage(tree, "Failed to process your approval decision"); + }); +}; + +/** + * Declines the blocked build. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + * @param {string} owner - The repository owner. + * @param {string} name - The repository name. + * @param {number} build - The build number. + */ +export const declineBuild = (tree, client, owner, repo, build) => { + client + .declineBuild(owner, repo, build) + .then(result => { + displayMessage(tree, "Successfully processed your decline decision"); + }) + .catch(() => { + displayMessage(tree, "Failed to process your decline decision"); + }); +}; + +/** + * Compare two builds by number. + * + * @param {Object} a - A build. + * @param {Object} b - A build. + * @returns {number} + */ +export const compareBuild = (a, b) => { + return b.number - a.number; +}; + +/** + * Returns true if the build is in a penidng or running state. + * + * @param {Object} build - The build object. + * @returns {boolean} + */ +export const assertBuildFinished = build => { + return build.status !== STATUS_RUNNING && build.status !== STATUS_PENDING; +}; + +/** + * Returns true if the build is a matrix. + * + * @param {Object} build - The build object. + * @returns {boolean} + */ +export const assertBuildMatrix = build => { + return build && build.procs && build.procs.length > 1; +}; diff --git a/web/src/shared/utils/feed.js b/web/src/shared/utils/feed.js new file mode 100644 index 000000000..5c542b59e --- /dev/null +++ b/web/src/shared/utils/feed.js @@ -0,0 +1,91 @@ +/** + * Get the event feed and store the results in the + * state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + */ +export const fetchFeed = (tree, client) => { + client + .getBuildFeed({ latest: true }) + .then(results => { + let list = {}; + let sorted = results.sort(compareFeedItem); + sorted.map(repo => { + list[repo.full_name] = repo; + }); + if (sorted && sorted.length > 0) { + tree.set(["feed", "latest"], sorted[0]); + } + tree.set(["feed", "loaded"], true); + tree.set(["feed", "data"], list); + }) + .catch(error => { + tree.set(["feed", "loaded"], true); + tree.set(["feed", "error"], error); + }); +}; + +/** + * Ensures the fetchFeed function is invoked exactly once. + * TODO replace this with a decorator + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + */ +export function fetchFeedOnce(tree, client) { + if (fetchFeedOnce.fired) { + return; + } + fetchFeedOnce.fired = true; + return fetchFeed(tree, client); +} + +/** + * Subscribes to the server-side event feed and synchonizes + * event data with the state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + */ +export const subscribeToFeed = (tree, client) => { + return client.on(data => { + const { repo, build } = data; + + if (tree.exists("feed", "data", repo.full_name)) { + const cursor = tree.select(["feed", "data", repo.full_name]); + cursor.merge(build); + } + + if (tree.exists("builds", "data", repo.full_name)) { + tree.set(["builds", "data", repo.full_name, build.number], build); + } + }); +}; + +/** + * Ensures the subscribeToFeed function is invoked exactly once. + * TODO replace this with a decorator + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + */ +export function subscribeToFeedOnce(tree, client) { + if (subscribeToFeedOnce.fired) { + return; + } + subscribeToFeedOnce.fired = true; + return subscribeToFeed(tree, client); +} + +/** + * Compare two feed items by name. + * @param {Object} a - A feed item. + * @param {Object} b - A feed item. + * @returns {number} + */ +export const compareFeedItem = (a, b) => { + return ( + (b.started_at || b.created_at || -1) - (a.started_at || a.created_at || -1) + ); +}; diff --git a/web/src/shared/utils/logs.js b/web/src/shared/utils/logs.js new file mode 100644 index 000000000..29020e301 --- /dev/null +++ b/web/src/shared/utils/logs.js @@ -0,0 +1,49 @@ +import { repositorySlug } from "./repository"; + +export function subscribeToLogs(tree, client, owner, repo, build, proc) { + if (subscribeToLogs.ws) { + subscribeToLogs.ws.close(); + } + const slug = repositorySlug(owner, repo); + const init = { data: [] }; + + tree.set(["logs", "data", slug, build, proc.pid], init); + + subscribeToLogs.ws = client.stream(owner, repo, build, proc.ppid, item => { + if (item.proc === proc.name) { + tree.push(["logs", "data", slug, build, proc.pid, "data"], item); + } + }); +} + +export function fetchLogs(tree, client, owner, repo, build, proc) { + const slug = repositorySlug(owner, repo); + const init = { + data: [], + loading: true, + }; + + tree.set(["logs", "data", slug, build, proc], init); + + client + .getLogs(owner, repo, build, proc) + .then(results => { + tree.set(["logs", "data", slug, build, proc, "data"], results || []); + tree.set(["logs", "data", slug, build, proc, "loading"], false); + tree.set(["logs", "data", slug, build, proc, "eof"], true); + }) + .catch(() => { + tree.set(["logs", "data", slug, build, proc, "loading"], false); + tree.set(["logs", "data", slug, build, proc, "eof"], true); + }); +} + +/** + * Toggles whether or not the browser should follow + * the logs (ie scroll to bottom). + * + * @param {boolean} follow - Follow the logs. + */ +export const toggleLogs = (tree, follow) => { + tree.set(["logs", "follow"], follow); +}; diff --git a/web/src/shared/utils/message.js b/web/src/shared/utils/message.js new file mode 100644 index 000000000..3f13312fc --- /dev/null +++ b/web/src/shared/utils/message.js @@ -0,0 +1,22 @@ +/** + * Displays the global message. + * + * @param {Object} tree - The drone state tree. + * @param {string} message - The message text. + */ +export const displayMessage = (tree, message) => { + tree.set(["message", "text"], message); + + setTimeout(() => { + hideMessage(tree); + }, 5000); +}; + +/** + * Hide the global message. + * + * @param {Object} tree - The drone state tree. + */ +export const hideMessage = tree => { + tree.unset(["message", "text"]); +}; diff --git a/web/src/shared/utils/proc.js b/web/src/shared/utils/proc.js new file mode 100644 index 000000000..cfa1cea01 --- /dev/null +++ b/web/src/shared/utils/proc.js @@ -0,0 +1,46 @@ +import { STATUS_PENDING, STATUS_RUNNING } from "shared/constants/status"; + +/** + * Returns a process from the process tree with the + * matching process number. + * + * @param {Object} procs - The process tree. + * @param {number|string} pid - The process number. + * @returns {Object} + */ +export const findChildProcess = (tree, pid) => { + for (var i = 0; i < tree.length; i++) { + const parent = tree[i]; + // eslint-disable-next-line + if (parent.pid == pid) { + return parent; + } + for (var ii = 0; ii < parent.children.length; ii++) { + const child = parent.children[ii]; + // eslint-disable-next-line + if (child.pid == pid) { + return child; + } + } + } +}; + +/** + * Returns true if the process is in a completed state. + * + * @param {Object} proc - The process object. + * @returns {boolean} + */ +export const assertProcFinished = proc => { + return proc.state !== STATUS_RUNNING && proc.state !== STATUS_PENDING; +}; + +/** + * Returns true if the process is running. + * + * @param {Object} proc - The process object. + * @returns {boolean} + */ +export const assertProcRunning = proc => { + return proc.state === STATUS_RUNNING; +}; diff --git a/web/src/shared/utils/registry.js b/web/src/shared/utils/registry.js new file mode 100644 index 000000000..94a61c860 --- /dev/null +++ b/web/src/shared/utils/registry.js @@ -0,0 +1,75 @@ +import { displayMessage } from "./message"; +import { repositorySlug } from "./repository"; + +/** + * Get the registry list for the named repository and + * store the results in the state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + * @param {string} owner - The repository owner. + * @param {string} name - The repository name. + */ +export const fetchRegistryList = (tree, client, owner, name) => { + const slug = repositorySlug(owner, name); + + tree.unset(["registry", "loaded"]); + tree.unset(["registry", "error"]); + + client.getRegistryList(owner, name).then(results => { + let list = {}; + results.map(registry => { + list[registry.address] = registry; + }); + tree.set(["registry", "data", slug], list); + tree.set(["registry", "loaded"], true); + }); +}; + +/** + * Create the registry credentials for the named repository + * and if successful, store the result in the state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + * @param {string} owner - The repository owner. + * @param {string} name - The repository name. + * @param {Object} registry - The registry hostname. + */ +export const createRegistry = (tree, client, owner, name, registry) => { + const slug = repositorySlug(owner, name); + + client + .createRegistry(owner, name, registry) + .then(result => { + tree.set(["registry", "data", slug, registry.address], result); + displayMessage(tree, "Successfully stored the registry credentials"); + }) + .catch(() => { + displayMessage(tree, "Failed to store the registry credentials"); + }); +}; + +/** + * Delete the registry credentials for the named repository + * and if successful, remove from the state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + * @param {string} owner - The repository owner. + * @param {string} name - The repository name. + * @param {Object} registry - The registry hostname. + */ +export const deleteRegistry = (tree, client, owner, name, registry) => { + const slug = repositorySlug(owner, name); + + client + .deleteRegistry(owner, name, registry) + .then(result => { + tree.unset(["registry", "data", slug, registry]); + displayMessage(tree, "Successfully deleted the registry credentials"); + }) + .catch(() => { + displayMessage(tree, "Failed to delete the registry credentials"); + }); +}; diff --git a/web/src/shared/utils/repository.js b/web/src/shared/utils/repository.js new file mode 100644 index 000000000..b8163b652 --- /dev/null +++ b/web/src/shared/utils/repository.js @@ -0,0 +1,186 @@ +import { displayMessage } from "./message"; +import { fetchFeed } from "shared/utils/feed"; + +/** + * Get the named repository and store the results in + * the state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + * @param {string} owner - The repository owner. + * @param {string} name - The repository name. + */ +export const fetchRepository = (tree, client, owner, name) => { + tree.unset(["repo", "error"]); + tree.unset(["repo", "loaded"]); + + client + .getRepo(owner, name) + .then(repo => { + tree.set(["repos", "data", repo.full_name], repo); + tree.set(["repo", "loaded"], true); + }) + .catch(error => { + tree.set(["repo", "error"], error); + tree.set(["repo", "loaded"], true); + }); +}; + +/** + * Get the repository list for the current user and + * store the results in the state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + */ +export const fetchRepostoryList = (tree, client) => { + tree.unset(["repos", "loaded"]); + tree.unset(["repos", "error"]); + + client + .getRepoList({ all: true }) + .then(results => { + let list = {}; + results.map(repo => { + list[repo.full_name] = repo; + }); + + const path = ["repos", "data"]; + if (tree.exists(path)) { + tree.deepMerge(path, list); + } else { + tree.set(path, list); + } + + tree.set(["repos", "loaded"], true); + }) + .catch(error => { + tree.set(["repos", "loaded"], true); + tree.set(["repos", "error"], error); + }); +}; + +/** + * Synchronize the repository list for the current user + * and merge the results into the state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + */ +export const syncRepostoryList = (tree, client) => { + tree.unset(["repos", "loaded"]); + tree.unset(["repos", "error"]); + + client + .getRepoList({ all: true, flush: true }) + .then(results => { + let list = {}; + results.map(repo => { + list[repo.full_name] = repo; + }); + + const path = ["repos", "data"]; + if (tree.exists(path)) { + tree.deepMerge(path, list); + } else { + tree.set(path, list); + } + + displayMessage(tree, "Successfully synchronized your repository list"); + tree.set(["repos", "loaded"], true); + }) + .catch(error => { + displayMessage(tree, "Failed to synchronize your repository list"); + tree.set(["repos", "loaded"], true); + tree.set(["repos", "error"], error); + }); +}; + +/** + * Update the repository and if successful update the + * repository information into the state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + * @param {string} owner - The repository owner. + * @param {string} name - The repository name. + * @param {Object} data - The repository updates. + */ +export const updateRepository = (tree, client, owner, name, data) => { + client + .updateRepo(owner, name, data) + .then(repo => { + tree.set(["repos", "data", repo.full_name], repo); + displayMessage(tree, "Successfully updated the repository settings"); + }) + .catch(() => { + displayMessage(tree, "Failed to update the repository settings"); + }); +}; + +/** + * Enables the repository and if successful update the + * repository active status in the state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + * @param {string} owner - The repository owner. + * @param {string} name - The repository name. + */ +export const enableRepository = (tree, client, owner, name) => { + client + .activateRepo(owner, name) + .then(result => { + displayMessage(tree, "Successfully activated your repository"); + tree.set(["repos", "data", result.full_name, "active"], true); + fetchFeed(tree, client); + }) + .catch(() => { + displayMessage(tree, "Failed to activate your repository"); + }); +}; + +/** + * Disables the repository and if successful update the + * repository active status in the state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + * @param {string} owner - The repository owner. + * @param {string} name - The repository name. + */ +export const disableRepository = (tree, client, owner, name) => { + client + .deleteRepo(owner, name) + .then(result => { + displayMessage(tree, "Successfully disabled your repository"); + tree.set(["repos", "data", result.full_name, "active"], false); + fetchFeed(tree, client); + }) + .catch(() => { + displayMessage(tree, "Failed to disabled your repository"); + }); +}; + +/** + * Compare two repositories by name. + * + * @param {Object} a - A repository. + * @param {Object} b - A repository. + * @returns {number} + */ +export const compareRepository = (a, b) => { + if (a.full_name < b.full_name) return -1; + if (a.full_name > b.full_name) return 1; + return 0; +}; + +/** + * Returns the repository slug. + * + * @param {string} owner - The repository owner. + * @param {string} name - The process name. + */ +export const repositorySlug = (owner, name) => { + return `${owner}/${name}`; +}; diff --git a/web/src/shared/utils/secrets.js b/web/src/shared/utils/secrets.js new file mode 100644 index 000000000..c28b34e7b --- /dev/null +++ b/web/src/shared/utils/secrets.js @@ -0,0 +1,75 @@ +import { displayMessage } from "./message"; +import { repositorySlug } from "./repository"; + +/** + * Get the secret list for the named repository and + * store the results in the state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + * @param {string} owner - The repository owner. + * @param {string} name - The repository name. + */ +export const fetchSecretList = (tree, client, owner, name) => { + const slug = repositorySlug(owner, name); + + tree.unset(["secrets", "loaded"]); + tree.unset(["secrets", "error"]); + + client.getSecretList(owner, name).then(results => { + let list = {}; + results.map(secret => { + list[secret.name] = secret; + }); + tree.set(["secrets", "data", slug], list); + tree.set(["secrets", "loaded"], true); + }); +}; + +/** + * Create the named repository secret and if successful + * store the result in the state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + * @param {string} owner - The repository owner. + * @param {string} name - The repository name. + * @param {Object} secret - The secret object. + */ +export const createSecret = (tree, client, owner, name, secret) => { + const slug = repositorySlug(owner, name); + + client + .createSecret(owner, name, secret) + .then(result => { + tree.set(["secrets", "data", slug, secret.name], result); + displayMessage(tree, "Successfully added the secret"); + }) + .catch(() => { + displayMessage(tree, "Failed to create the secret"); + }); +}; + +/** + * Delete the named repository secret from the server and + * remove from the state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + * @param {string} owner - The repository owner. + * @param {string} name - The repository name. + * @param {string} secret - The secret name. + */ +export const deleteSecret = (tree, client, owner, name, secret) => { + const slug = repositorySlug(owner, name); + + client + .deleteSecret(owner, name, secret) + .then(result => { + tree.unset(["secrets", "data", slug, secret]); + displayMessage(tree, "Successfully removed the secret"); + }) + .catch(() => { + displayMessage(tree, "Failed to remove the secret"); + }); +}; diff --git a/web/src/shared/utils/users.js b/web/src/shared/utils/users.js new file mode 100644 index 000000000..925c28253 --- /dev/null +++ b/web/src/shared/utils/users.js @@ -0,0 +1,19 @@ +import { displayMessage } from "./message"; + +/** +* Generates a personal access token and stores the results in +* the state tree. + * + * @param {Object} tree - The drone state tree. + * @param {Object} client - The drone client. + */ +export const generateToken = (tree, client) => { + client + .getToken() + .then(token => { + tree.set(["token"], token); + }) + .catch(() => { + displayMessage(tree, "Failed to retrieve your personal access token"); + }); +}; diff --git a/web/vendor/drone-js/index.js b/web/vendor/drone-js/index.js new file mode 100644 index 000000000..e7106b2a2 --- /dev/null +++ b/web/vendor/drone-js/index.js @@ -0,0 +1,306 @@ +(function (global, factory) { + if (typeof define === "function" && define.amd) { + define(["exports"], factory); + } else if (typeof exports !== "undefined") { + factory(exports); + } else { + var mod = { + exports: {} + }; + factory(mod.exports); + global.index = mod.exports; + } +})(this, function (exports) { + "use strict"; + + Object.defineProperty(exports, "__esModule", { + value: true + }); + + function _classCallCheck(instance, Constructor) { + if (!(instance instanceof Constructor)) { + throw new TypeError("Cannot call a class as a function"); + } + } + + var _createClass = function () { + function defineProperties(target, props) { + for (var i = 0; i < props.length; i++) { + var descriptor = props[i]; + descriptor.enumerable = descriptor.enumerable || false; + descriptor.configurable = true; + if ("value" in descriptor) descriptor.writable = true; + Object.defineProperty(target, descriptor.key, descriptor); + } + } + + return function (Constructor, protoProps, staticProps) { + if (protoProps) defineProperties(Constructor.prototype, protoProps); + if (staticProps) defineProperties(Constructor, staticProps); + return Constructor; + }; + }(); + + var DroneClient = function () { + function DroneClient(server, token, csrf) { + _classCallCheck(this, DroneClient); + + this.server = server || ""; + this.token = token; + this.csrf = csrf; + } + + _createClass(DroneClient, [{ + key: "getRepoList", + value: function getRepoList(opts) { + var query = encodeQueryString(opts); + return this._get("/api/user/repos?" + query); + } + }, { + key: "getRepo", + value: function getRepo(owner, repo) { + return this._get("/api/repos/" + owner + "/" + repo); + } + }, { + key: "activateRepo", + value: function activateRepo(owner, repo) { + return this._post("/api/repos/" + owner + "/" + repo); + } + }, { + key: "updateRepo", + value: function updateRepo(owner, repo, data) { + return this._patch("/api/repos/" + owner + "/" + repo, data); + } + }, { + key: "deleteRepo", + value: function deleteRepo(owner, repo) { + return this._delete("/api/repos/" + owner + "/" + repo); + } + }, { + key: "getBuildList", + value: function getBuildList(owner, repo, opts) { + var query = encodeQueryString(opts); + return this._get("/api/repos/" + owner + "/" + repo + "/builds?" + query); + } + }, { + key: "getBuild", + value: function getBuild(owner, repo, number) { + return this._get("/api/repos/" + owner + "/" + repo + "/builds/" + number); + } + }, { + key: "getBuildFeed", + value: function getBuildFeed(opts) { + var query = encodeQueryString(opts); + return this._get("/api/user/feed?" + query); + } + }, { + key: "cancelBuild", + value: function cancelBuild(owner, repo, number, ppid) { + return this._delete("/api/repos/" + owner + "/" + repo + "/builds/" + number + "/" + ppid); + } + }, { + key: "approveBuild", + value: function approveBuild(owner, repo, build) { + return this._post("/api/repos/" + owner + "/" + repo + "/builds/" + build + "/approve"); + } + }, { + key: "declineBuild", + value: function declineBuild(owner, repo, build) { + return this._post("/api/repos/" + owner + "/" + repo + "/builds/" + build + "/decline"); + } + }, { + key: "restartBuild", + value: function restartBuild(owner, repo, build, opts) { + var query = encodeQueryString(opts); + return this._post("/api/repos/" + owner + "/" + repo + "/builds/" + build + "?" + query); + } + }, { + key: "getLogs", + value: function getLogs(owner, repo, build, proc) { + return this._get("/api/repos/" + owner + "/" + repo + "/logs/" + build + "/" + proc); + } + }, { + key: "getArtifact", + value: function getArtifact(owner, repo, build, proc, file) { + return this._get("/api/repos/" + owner + "/" + repo + "/files/" + build + "/" + proc + "/" + file + "?raw=true"); + } + }, { + key: "getArtifactList", + value: function getArtifactList(owner, repo, build) { + return this._get("/api/repos/" + owner + "/" + repo + "/files/" + build); + } + }, { + key: "getSecretList", + value: function getSecretList(owner, repo) { + return this._get("/api/repos/" + owner + "/" + repo + "/secrets"); + } + }, { + key: "createSecret", + value: function createSecret(owner, repo, secret) { + return this._post("/api/repos/" + owner + "/" + repo + "/secrets", secret); + } + }, { + key: "deleteSecret", + value: function deleteSecret(owner, repo, secret) { + return this._delete("/api/repos/" + owner + "/" + repo + "/secrets/" + secret); + } + }, { + key: "getRegistryList", + value: function getRegistryList(owner, repo) { + return this._get("/api/repos/" + owner + "/" + repo + "/registry"); + } + }, { + key: "createRegistry", + value: function createRegistry(owner, repo, registry) { + return this._post("/api/repos/" + owner + "/" + repo + "/registry", registry); + } + }, { + key: "deleteRegistry", + value: function deleteRegistry(owner, repo, address) { + return this._delete("/api/repos/" + owner + "/" + repo + "/registry/" + address); + } + }, { + key: "getSelf", + value: function getSelf() { + return this._get("/api/user"); + } + }, { + key: "getToken", + value: function getToken() { + return this._post("/api/user/token"); + } + }, { + key: "on", + value: function on(callback) { + return this._subscribe("/stream/events", callback, { + reconnect: true + }); + } + }, { + key: "stream", + value: function stream(owner, repo, build, proc, callback) { + return this._subscribe("/stream/logs/" + owner + "/" + repo + "/" + build + "/" + proc, callback, { + reconnect: false + }); + } + }, { + key: "_get", + value: function _get(path) { + return this._request("GET", path, null); + } + }, { + key: "_post", + value: function _post(path, data) { + return this._request("POST", path, data); + } + }, { + key: "_patch", + value: function _patch(path, data) { + return this._request("PATCH", path, data); + } + }, { + key: "_delete", + value: function _delete(path) { + return this._request("DELETE", path, null); + } + }, { + key: "_subscribe", + value: function _subscribe(path, callback, opts) { + var query = encodeQueryString({ + access_token: this.token + }); + path = this.server ? this.server + path : path; + path = this.token ? path + "?" + query : path; + + var events = new EventSource(path); + events.onmessage = function (event) { + var data = JSON.parse(event.data); + callback(data); + }; + if (!opts.reconnect) { + events.onerror = function (err) { + if (err.data === "eof") { + events.close(); + } + }; + } + return events; + } + }, { + key: "_request", + value: function _request(method, path, data) { + var endpoint = [this.server, path].join(""); + var xhr = new XMLHttpRequest(); + xhr.open(method, endpoint, true); + if (this.token) { + xhr.setRequestHeader("Authorization", "Bearer " + this.token); + } + if (method !== "GET" && this.csrf) { + xhr.setRequestHeader("X-CSRF-TOKEN", this.csrf); + } + return new Promise(function (resolve, reject) { + xhr.onload = function () { + if (xhr.readyState === 4) { + if (xhr.status >= 300) { + var error = { + status: xhr.status, + message: xhr.response + }; + if (this.onerror) { + this.onerror(error); + } + reject(error); + return; + } + var contentType = xhr.getResponseHeader("Content-Type"); + if (contentType && contentType.startsWith("application/json")) { + resolve(JSON.parse(xhr.response)); + } else { + resolve(xhr.response); + } + } + }.bind(this); + xhr.onerror = function (e) { + reject(e); + }; + if (data) { + xhr.setRequestHeader("Content-Type", "application/json"); + xhr.send(JSON.stringify(data)); + } else { + xhr.send(); + } + }.bind(this)); + } + }], [{ + key: "fromEnviron", + value: function fromEnviron() { + return new DroneClient(process && process.env && process.env.DRONE_SERVER, process && process.env && process.env.DRONE_TOKEN, process && process.env && process.env.DRONE_CSRF); + } + }, { + key: "fromWindow", + value: function fromWindow() { + return new DroneClient(window && window.DRONE_SERVER, window && window.DRONE_TOKEN, window && window.DRONE_CSRF); + } + }]); + + return DroneClient; + }(); + + exports.default = DroneClient; + + + /** + * Encodes the values into url encoded form sorted by key. + * + * @param {object} query parameters in key value object. + * @return {string} query parameter string + */ + var encodeQueryString = exports.encodeQueryString = function encodeQueryString() { + var params = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; + + return params ? Object.keys(params).sort().map(function (key) { + var val = params[key]; + return encodeURIComponent(key) + "=" + encodeURIComponent(val); + }).join("&") : ""; + }; +}); \ No newline at end of file diff --git a/web/webpack.config.js b/web/webpack.config.js new file mode 100644 index 000000000..cf913d162 --- /dev/null +++ b/web/webpack.config.js @@ -0,0 +1,160 @@ +require("dotenv").config(); + +var path = require("path"); +var webpack = require("webpack"); +var HtmlWebpackPlugin = require("html-webpack-plugin"); + +const ENV = process.env.NODE_ENV || "development"; + +module.exports = { + entry: { + app: "./src", + vendor: [ + "ansi_up", + "babel-polyfill", + "baobab", + "baobab-react", + "classnames", + "drone-js", + "humanize-duration", + "preact", + "preact-compat", + "query-string", + "react-router", + "react-router-dom", + "react-screen-size", + "react-timeago", + "react-title-component", + "react-transition-group" + ] + }, + + // where to dump the output of a production build + output: { + publicPath: "/", + path: path.join(__dirname, "dist/files"), + filename: "static/bundle.[chunkhash].js" + }, + + resolve: { + alias: { + client: path.resolve(__dirname, "src/client/"), + config: path.resolve(__dirname, "src/config/"), + components: path.resolve(__dirname, "src/components/"), + layouts: path.resolve(__dirname, "src/layouts/"), + pages: path.resolve(__dirname, "src/pages/"), + screens: path.resolve(__dirname, "src/screens/"), + shared: path.resolve(__dirname, "src/shared/"), + + react: "preact-compat/dist/preact-compat", + "react-dom": "preact-compat/dist/preact-compat", + "create-react-class": "preact-compat/lib/create-react-class" + } + }, + + module: { + rules: [ + { + test: /\.jsx?/i, + exclude: /node_modules/, + loader: "babel-loader" + }, + + { + test: /\.(less|css)$/, + loader: "style-loader" + }, + + { + test: /\.(less|css)$/, + loader: "css-loader", + query: { + modules: true, + localIdentName: "[name]__[local]___[hash:base64:5]" + } + }, + + { + test: /\.(less|css)$/, + loader: "less-loader" + } + ] + }, + + plugins: [ + new webpack.optimize.CommonsChunkPlugin({ + name: "vendor", + filename: "static/vendor.[hash].js" + }), + new HtmlWebpackPlugin({ + favicon: "src/public/favicon.png", + template: "src/index.html" + }) + ].concat( + ENV === "production" + ? [ + new webpack.optimize.UglifyJsPlugin({ + output: { + comments: false + }, + exclude: [/bundle/], + compress: { + unsafe_comps: true, + properties: true, + keep_fargs: false, + pure_getters: true, + collapse_vars: true, + unsafe: true, + warnings: false, + screw_ie8: true, + sequences: true, + dead_code: true, + drop_debugger: true, + comparisons: true, + conditionals: true, + evaluate: true, + booleans: true, + loops: true, + unused: true, + hoist_funs: true, + if_return: true, + join_vars: true, + cascade: true, + drop_console: true + } + }) + ] + : [ + new webpack.DefinePlugin({ + // drone server uses authorization cookies, but the client can + // optionally source the authorization token from the environment. + // this should be used for the test server only. + "window.DRONE_TOKEN": JSON.stringify(process.env.DRONE_TOKEN), + "window.DRONE_SERVER": JSON.stringify(process.env.DRONE_SERVER), + + // drone server provides the currently authenticated user in the + // index.html file. For testing purposes we simulate this and provides + // a dummy user object. + "window.DRONE_USER": { + login: JSON.stringify("octocat"), + avatar_url: JSON.stringify( + "https://avatars3.githubusercontent.com/u/583231" + ) + } + }) + ] + ), + + devServer: { + port: process.env.PORT || 9999, + + // serve up any static files from src/ + contentBase: path.join(__dirname, "src"), + + // enable gzip compression: + compress: true, + + // enable pushState() routing, as used by preact-router et al: + historyApiFallback: true + } +}; diff --git a/web/yarn.lock b/web/yarn.lock new file mode 100644 index 000000000..b60be10dc --- /dev/null +++ b/web/yarn.lock @@ -0,0 +1,6604 @@ +# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. +# yarn lockfile v1 + + +abab@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/abab/-/abab-1.0.3.tgz#b81de5f7274ec4e756d797cd834f303642724e5d" + +abbrev@1: + version "1.1.0" + resolved "https://registry.yarnpkg.com/abbrev/-/abbrev-1.1.0.tgz#d0554c2256636e2f56e7c2e5ad183f859428d81f" + +accepts@~1.3.3: + version "1.3.4" + resolved "https://registry.yarnpkg.com/accepts/-/accepts-1.3.4.tgz#86246758c7dd6d21a6474ff084a4740ec05eb21f" + dependencies: + mime-types "~2.1.16" + negotiator "0.6.1" + +acorn-dynamic-import@^2.0.0: + version "2.0.2" + resolved "https://registry.yarnpkg.com/acorn-dynamic-import/-/acorn-dynamic-import-2.0.2.tgz#c752bd210bef679501b6c6cb7fc84f8f47158cc4" + dependencies: + acorn "^4.0.3" + +acorn-globals@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/acorn-globals/-/acorn-globals-3.1.0.tgz#fd8270f71fbb4996b004fa880ee5d46573a731bf" + dependencies: + acorn "^4.0.4" + +acorn-jsx@^3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/acorn-jsx/-/acorn-jsx-3.0.1.tgz#afdf9488fb1ecefc8348f6fb22f464e32a58b36b" + dependencies: + acorn "^3.0.4" + +acorn@^3.0.4: + version "3.3.0" + resolved "https://registry.yarnpkg.com/acorn/-/acorn-3.3.0.tgz#45e37fb39e8da3f25baee3ff5369e2bb5f22017a" + +acorn@^4.0.3, acorn@^4.0.4: + version "4.0.13" + resolved "https://registry.yarnpkg.com/acorn/-/acorn-4.0.13.tgz#105495ae5361d697bd195c825192e1ad7f253787" + +acorn@^5.0.0, acorn@^5.1.1: + version "5.1.2" + resolved "https://registry.yarnpkg.com/acorn/-/acorn-5.1.2.tgz#911cb53e036807cf0fa778dc5d370fbd864246d7" + +add-matchers@0.5.0: + version "0.5.0" + resolved "https://registry.yarnpkg.com/add-matchers/-/add-matchers-0.5.0.tgz#502190e4750cd5721618393268b61a157366e765" + +ajv-keywords@^1.0.0: + version "1.5.1" + resolved "https://registry.yarnpkg.com/ajv-keywords/-/ajv-keywords-1.5.1.tgz#314dd0a4b3368fad3dfcdc54ede6171b886daf3c" + +ajv-keywords@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/ajv-keywords/-/ajv-keywords-2.1.0.tgz#a296e17f7bfae7c1ce4f7e0de53d29cb32162df0" + +ajv@^4.7.0, ajv@^4.9.1: + version "4.11.8" + resolved "https://registry.yarnpkg.com/ajv/-/ajv-4.11.8.tgz#82ffb02b29e662ae53bdc20af15947706739c536" + dependencies: + co "^4.6.0" + json-stable-stringify "^1.0.1" + +ajv@^5.0.0, ajv@^5.1.5, ajv@^5.2.0: + version "5.2.2" + resolved "https://registry.yarnpkg.com/ajv/-/ajv-5.2.2.tgz#47c68d69e86f5d953103b0074a9430dc63da5e39" + dependencies: + co "^4.6.0" + fast-deep-equal "^1.0.0" + json-schema-traverse "^0.3.0" + json-stable-stringify "^1.0.1" + +align-text@^0.1.1, align-text@^0.1.3: + version "0.1.4" + resolved "https://registry.yarnpkg.com/align-text/-/align-text-0.1.4.tgz#0cd90a561093f35d0a99256c22b7069433fad117" + dependencies: + kind-of "^3.0.2" + longest "^1.0.1" + repeat-string "^1.5.2" + +alphanum-sort@^1.0.1, alphanum-sort@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/alphanum-sort/-/alphanum-sort-1.0.2.tgz#97a1119649b211ad33691d9f9f486a8ec9fbe0a3" + +ansi-escapes@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/ansi-escapes/-/ansi-escapes-2.0.0.tgz#5bae52be424878dd9783e8910e3fc2922e83c81b" + +ansi-escapes@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/ansi-escapes/-/ansi-escapes-3.0.0.tgz#ec3e8b4e9f8064fc02c3ac9b65f1c275bda8ef92" + +ansi-html@0.0.7: + version "0.0.7" + resolved "https://registry.yarnpkg.com/ansi-html/-/ansi-html-0.0.7.tgz#813584021962a9e9e6fd039f940d12f56ca7859e" + +ansi-regex@^2.0.0: + version "2.1.1" + resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-2.1.1.tgz#c3b33ab5ee360d86e0e628f0468ae7ef27d654df" + +ansi-regex@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-3.0.0.tgz#ed0317c322064f79466c02966bddb605ab37d998" + +ansi-styles@^2.2.1: + version "2.2.1" + resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-2.2.1.tgz#b432dd3358b634cf75e1e4664368240533c1ddbe" + +ansi-styles@^3.1.0, ansi-styles@^3.2.0: + version "3.2.0" + resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-3.2.0.tgz#c159b8d5be0f9e5a6f346dab94f16ce022161b88" + dependencies: + color-convert "^1.9.0" + +ansi_up@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/ansi_up/-/ansi_up-2.0.2.tgz#9b54de508c5c579f5b6968e65c1b863e0680ab92" + +anymatch@^1.3.0: + version "1.3.2" + resolved "https://registry.yarnpkg.com/anymatch/-/anymatch-1.3.2.tgz#553dcb8f91e3c889845dfdba34c77721b90b9d7a" + dependencies: + micromatch "^2.1.5" + normalize-path "^2.0.0" + +append-transform@^0.4.0: + version "0.4.0" + resolved "https://registry.yarnpkg.com/append-transform/-/append-transform-0.4.0.tgz#d76ebf8ca94d276e247a36bad44a4b74ab611991" + dependencies: + default-require-extensions "^1.0.0" + +aproba@^1.0.3: + version "1.1.2" + resolved "https://registry.yarnpkg.com/aproba/-/aproba-1.1.2.tgz#45c6629094de4e96f693ef7eab74ae079c240fc1" + +are-we-there-yet@~1.1.2: + version "1.1.4" + resolved "https://registry.yarnpkg.com/are-we-there-yet/-/are-we-there-yet-1.1.4.tgz#bb5dca382bb94f05e15194373d16fd3ba1ca110d" + dependencies: + delegates "^1.0.0" + readable-stream "^2.0.6" + +argparse@^1.0.7: + version "1.0.9" + resolved "https://registry.yarnpkg.com/argparse/-/argparse-1.0.9.tgz#73d83bc263f86e97f8cc4f6bae1b0e90a7d22c86" + dependencies: + sprintf-js "~1.0.2" + +arr-diff@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/arr-diff/-/arr-diff-2.0.0.tgz#8f3b827f955a8bd669697e4a4256ac3ceae356cf" + dependencies: + arr-flatten "^1.0.1" + +arr-flatten@^1.0.1: + version "1.1.0" + resolved "https://registry.yarnpkg.com/arr-flatten/-/arr-flatten-1.1.0.tgz#36048bbff4e7b47e136644316c99669ea5ae91f1" + +array-equal@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/array-equal/-/array-equal-1.0.0.tgz#8c2a5ef2472fd9ea742b04c77a75093ba2757c93" + +array-find-index@^1.0.1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/array-find-index/-/array-find-index-1.0.2.tgz#df010aa1287e164bbda6f9723b0a96a1ec4187a1" + +array-flatten@1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/array-flatten/-/array-flatten-1.1.1.tgz#9a5f699051b1e7073328f2a008968b64ea2955d2" + +array-flatten@^2.1.0: + version "2.1.1" + resolved "https://registry.yarnpkg.com/array-flatten/-/array-flatten-2.1.1.tgz#426bb9da84090c1838d812c8150af20a8331e296" + +array-includes@^3.0.3: + version "3.0.3" + resolved "https://registry.yarnpkg.com/array-includes/-/array-includes-3.0.3.tgz#184b48f62d92d7452bb31b323165c7f8bd02266d" + dependencies: + define-properties "^1.1.2" + es-abstract "^1.7.0" + +array-union@^1.0.1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/array-union/-/array-union-1.0.2.tgz#9a34410e4f4e3da23dea375be5be70f24778ec39" + dependencies: + array-uniq "^1.0.1" + +array-uniq@^1.0.1: + version "1.0.3" + resolved "https://registry.yarnpkg.com/array-uniq/-/array-uniq-1.0.3.tgz#af6ac877a25cc7f74e058894753858dfdb24fdb6" + +array-unique@^0.2.1: + version "0.2.1" + resolved "https://registry.yarnpkg.com/array-unique/-/array-unique-0.2.1.tgz#a1d97ccafcbc2625cc70fadceb36a50c58b01a53" + +arrify@^1.0.0, arrify@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/arrify/-/arrify-1.0.1.tgz#898508da2226f380df904728456849c1501a4b0d" + +asap@~2.0.3: + version "2.0.6" + resolved "https://registry.yarnpkg.com/asap/-/asap-2.0.6.tgz#e50347611d7e690943208bbdafebcbc2fb866d46" + +asn1.js@^4.0.0: + version "4.9.1" + resolved "https://registry.yarnpkg.com/asn1.js/-/asn1.js-4.9.1.tgz#48ba240b45a9280e94748990ba597d216617fd40" + dependencies: + bn.js "^4.0.0" + inherits "^2.0.1" + minimalistic-assert "^1.0.0" + +asn1@~0.2.3: + version "0.2.3" + resolved "https://registry.yarnpkg.com/asn1/-/asn1-0.2.3.tgz#dac8787713c9966849fc8180777ebe9c1ddf3b86" + +assert-plus@1.0.0, assert-plus@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/assert-plus/-/assert-plus-1.0.0.tgz#f12e0f3c5d77b0b1cdd9146942e4e96c1e4dd525" + +assert-plus@^0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/assert-plus/-/assert-plus-0.2.0.tgz#d74e1b87e7affc0db8aadb7021f3fe48101ab234" + +assert@^1.1.1: + version "1.4.1" + resolved "https://registry.yarnpkg.com/assert/-/assert-1.4.1.tgz#99912d591836b5a6f5b345c0f07eefc08fc65d91" + dependencies: + util "0.10.3" + +astral-regex@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/astral-regex/-/astral-regex-1.0.0.tgz#6c8c3fb827dd43ee3918f27b82782ab7658a6fd9" + +async-each@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/async-each/-/async-each-1.0.1.tgz#19d386a1d9edc6e7c1c85d388aedbcc56d33602d" + +async@^1.5.2: + version "1.5.2" + resolved "https://registry.yarnpkg.com/async/-/async-1.5.2.tgz#ec6a61ae56480c0c3cb241c95618e20892f9672a" + +async@^2.1.2, async@^2.1.4: + version "2.5.0" + resolved "https://registry.yarnpkg.com/async/-/async-2.5.0.tgz#843190fd6b7357a0b9e1c956edddd5ec8462b54d" + dependencies: + lodash "^4.14.0" + +asynckit@^0.4.0: + version "0.4.0" + resolved "https://registry.yarnpkg.com/asynckit/-/asynckit-0.4.0.tgz#c79ed97f7f34cb8f2ba1bc9790bcc366474b4b79" + +autoprefixer@^6.3.1: + version "6.7.7" + resolved "https://registry.yarnpkg.com/autoprefixer/-/autoprefixer-6.7.7.tgz#1dbd1c835658e35ce3f9984099db00585c782014" + dependencies: + browserslist "^1.7.6" + caniuse-db "^1.0.30000634" + normalize-range "^0.1.2" + num2fraction "^1.2.2" + postcss "^5.2.16" + postcss-value-parser "^3.2.3" + +aws-sign2@~0.6.0: + version "0.6.0" + resolved "https://registry.yarnpkg.com/aws-sign2/-/aws-sign2-0.6.0.tgz#14342dd38dbcc94d0e5b87d763cd63612c0e794f" + +aws4@^1.2.1: + version "1.6.0" + resolved "https://registry.yarnpkg.com/aws4/-/aws4-1.6.0.tgz#83ef5ca860b2b32e4a0deedee8c771b9db57471e" + +babel-code-frame@^6.11.0, babel-code-frame@^6.22.0, babel-code-frame@^6.26.0: + version "6.26.0" + resolved "https://registry.yarnpkg.com/babel-code-frame/-/babel-code-frame-6.26.0.tgz#63fd43f7dc1e3bb7ce35947db8fe369a3f58c74b" + dependencies: + chalk "^1.1.3" + esutils "^2.0.2" + js-tokens "^3.0.2" + +babel-core@^6.0.0, babel-core@^6.25.0, babel-core@^6.26.0: + version "6.26.0" + resolved "https://registry.yarnpkg.com/babel-core/-/babel-core-6.26.0.tgz#af32f78b31a6fcef119c87b0fd8d9753f03a0bb8" + dependencies: + babel-code-frame "^6.26.0" + babel-generator "^6.26.0" + babel-helpers "^6.24.1" + babel-messages "^6.23.0" + babel-register "^6.26.0" + babel-runtime "^6.26.0" + babel-template "^6.26.0" + babel-traverse "^6.26.0" + babel-types "^6.26.0" + babylon "^6.18.0" + convert-source-map "^1.5.0" + debug "^2.6.8" + json5 "^0.5.1" + lodash "^4.17.4" + minimatch "^3.0.4" + path-is-absolute "^1.0.1" + private "^0.1.7" + slash "^1.0.0" + source-map "^0.5.6" + +babel-eslint: + version "7.2.3" + resolved "https://registry.yarnpkg.com/babel-eslint/-/babel-eslint-7.2.3.tgz#b2fe2d80126470f5c19442dc757253a897710827" + dependencies: + babel-code-frame "^6.22.0" + babel-traverse "^6.23.1" + babel-types "^6.23.0" + babylon "^6.17.0" + +babel-generator@^6.18.0, babel-generator@^6.26.0: + version "6.26.0" + resolved "https://registry.yarnpkg.com/babel-generator/-/babel-generator-6.26.0.tgz#ac1ae20070b79f6e3ca1d3269613053774f20dc5" + dependencies: + babel-messages "^6.23.0" + babel-runtime "^6.26.0" + babel-types "^6.26.0" + detect-indent "^4.0.0" + jsesc "^1.3.0" + lodash "^4.17.4" + source-map "^0.5.6" + trim-right "^1.0.1" + +babel-helper-bindify-decorators@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-helper-bindify-decorators/-/babel-helper-bindify-decorators-6.24.1.tgz#14c19e5f142d7b47f19a52431e52b1ccbc40a330" + dependencies: + babel-runtime "^6.22.0" + babel-traverse "^6.24.1" + babel-types "^6.24.1" + +babel-helper-builder-binary-assignment-operator-visitor@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-helper-builder-binary-assignment-operator-visitor/-/babel-helper-builder-binary-assignment-operator-visitor-6.24.1.tgz#cce4517ada356f4220bcae8a02c2b346f9a56664" + dependencies: + babel-helper-explode-assignable-expression "^6.24.1" + babel-runtime "^6.22.0" + babel-types "^6.24.1" + +babel-helper-builder-react-jsx@^6.24.1: + version "6.26.0" + resolved "https://registry.yarnpkg.com/babel-helper-builder-react-jsx/-/babel-helper-builder-react-jsx-6.26.0.tgz#39ff8313b75c8b65dceff1f31d383e0ff2a408a0" + dependencies: + babel-runtime "^6.26.0" + babel-types "^6.26.0" + esutils "^2.0.2" + +babel-helper-call-delegate@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-helper-call-delegate/-/babel-helper-call-delegate-6.24.1.tgz#ece6aacddc76e41c3461f88bfc575bd0daa2df8d" + dependencies: + babel-helper-hoist-variables "^6.24.1" + babel-runtime "^6.22.0" + babel-traverse "^6.24.1" + babel-types "^6.24.1" + +babel-helper-define-map@^6.24.1: + version "6.26.0" + resolved "https://registry.yarnpkg.com/babel-helper-define-map/-/babel-helper-define-map-6.26.0.tgz#a5f56dab41a25f97ecb498c7ebaca9819f95be5f" + dependencies: + babel-helper-function-name "^6.24.1" + babel-runtime "^6.26.0" + babel-types "^6.26.0" + lodash "^4.17.4" + +babel-helper-explode-assignable-expression@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-helper-explode-assignable-expression/-/babel-helper-explode-assignable-expression-6.24.1.tgz#f25b82cf7dc10433c55f70592d5746400ac22caa" + dependencies: + babel-runtime "^6.22.0" + babel-traverse "^6.24.1" + babel-types "^6.24.1" + +babel-helper-explode-class@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-helper-explode-class/-/babel-helper-explode-class-6.24.1.tgz#7dc2a3910dee007056e1e31d640ced3d54eaa9eb" + dependencies: + babel-helper-bindify-decorators "^6.24.1" + babel-runtime "^6.22.0" + babel-traverse "^6.24.1" + babel-types "^6.24.1" + +babel-helper-function-name@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-helper-function-name/-/babel-helper-function-name-6.24.1.tgz#d3475b8c03ed98242a25b48351ab18399d3580a9" + dependencies: + babel-helper-get-function-arity "^6.24.1" + babel-runtime "^6.22.0" + babel-template "^6.24.1" + babel-traverse "^6.24.1" + babel-types "^6.24.1" + +babel-helper-get-function-arity@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-helper-get-function-arity/-/babel-helper-get-function-arity-6.24.1.tgz#8f7782aa93407c41d3aa50908f89b031b1b6853d" + dependencies: + babel-runtime "^6.22.0" + babel-types "^6.24.1" + +babel-helper-hoist-variables@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-helper-hoist-variables/-/babel-helper-hoist-variables-6.24.1.tgz#1ecb27689c9d25513eadbc9914a73f5408be7a76" + dependencies: + babel-runtime "^6.22.0" + babel-types "^6.24.1" + +babel-helper-optimise-call-expression@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-helper-optimise-call-expression/-/babel-helper-optimise-call-expression-6.24.1.tgz#f7a13427ba9f73f8f4fa993c54a97882d1244257" + dependencies: + babel-runtime "^6.22.0" + babel-types "^6.24.1" + +babel-helper-regex@^6.24.1: + version "6.26.0" + resolved "https://registry.yarnpkg.com/babel-helper-regex/-/babel-helper-regex-6.26.0.tgz#325c59f902f82f24b74faceed0363954f6495e72" + dependencies: + babel-runtime "^6.26.0" + babel-types "^6.26.0" + lodash "^4.17.4" + +babel-helper-remap-async-to-generator@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-helper-remap-async-to-generator/-/babel-helper-remap-async-to-generator-6.24.1.tgz#5ec581827ad723fecdd381f1c928390676e4551b" + dependencies: + babel-helper-function-name "^6.24.1" + babel-runtime "^6.22.0" + babel-template "^6.24.1" + babel-traverse "^6.24.1" + babel-types "^6.24.1" + +babel-helper-replace-supers@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-helper-replace-supers/-/babel-helper-replace-supers-6.24.1.tgz#bf6dbfe43938d17369a213ca8a8bf74b6a90ab1a" + dependencies: + babel-helper-optimise-call-expression "^6.24.1" + babel-messages "^6.23.0" + babel-runtime "^6.22.0" + babel-template "^6.24.1" + babel-traverse "^6.24.1" + babel-types "^6.24.1" + +babel-helpers@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-helpers/-/babel-helpers-6.24.1.tgz#3471de9caec388e5c850e597e58a26ddf37602b2" + dependencies: + babel-runtime "^6.22.0" + babel-template "^6.24.1" + +babel-jest@^21.0.0, babel-jest@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/babel-jest/-/babel-jest-21.0.2.tgz#817ea52c23f1c6c4b684d6960968416b6a9e9c6c" + dependencies: + babel-plugin-istanbul "^4.0.0" + babel-preset-jest "^21.0.2" + +babel-loader@^7.1.1: + version "7.1.2" + resolved "https://registry.yarnpkg.com/babel-loader/-/babel-loader-7.1.2.tgz#f6cbe122710f1aa2af4d881c6d5b54358ca24126" + dependencies: + find-cache-dir "^1.0.0" + loader-utils "^1.0.2" + mkdirp "^0.5.1" + +babel-messages@^6.23.0: + version "6.23.0" + resolved "https://registry.yarnpkg.com/babel-messages/-/babel-messages-6.23.0.tgz#f3cdf4703858035b2a2951c6ec5edf6c62f2630e" + dependencies: + babel-runtime "^6.22.0" + +babel-plugin-check-es2015-constants@^6.22.0: + version "6.22.0" + resolved "https://registry.yarnpkg.com/babel-plugin-check-es2015-constants/-/babel-plugin-check-es2015-constants-6.22.0.tgz#35157b101426fd2ffd3da3f75c7d1e91835bbf8a" + dependencies: + babel-runtime "^6.22.0" + +babel-plugin-istanbul@^4.0.0: + version "4.1.4" + resolved "https://registry.yarnpkg.com/babel-plugin-istanbul/-/babel-plugin-istanbul-4.1.4.tgz#18dde84bf3ce329fddf3f4103fae921456d8e587" + dependencies: + find-up "^2.1.0" + istanbul-lib-instrument "^1.7.2" + test-exclude "^4.1.1" + +babel-plugin-jest-hoist@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/babel-plugin-jest-hoist/-/babel-plugin-jest-hoist-21.0.2.tgz#cfdce5bca40d772a056cb8528ad159c7bb4bb03d" + +babel-plugin-syntax-async-functions@^6.8.0: + version "6.13.0" + resolved "https://registry.yarnpkg.com/babel-plugin-syntax-async-functions/-/babel-plugin-syntax-async-functions-6.13.0.tgz#cad9cad1191b5ad634bf30ae0872391e0647be95" + +babel-plugin-syntax-async-generators@^6.5.0: + version "6.13.0" + resolved "https://registry.yarnpkg.com/babel-plugin-syntax-async-generators/-/babel-plugin-syntax-async-generators-6.13.0.tgz#6bc963ebb16eccbae6b92b596eb7f35c342a8b9a" + +babel-plugin-syntax-class-constructor-call@^6.18.0: + version "6.18.0" + resolved "https://registry.yarnpkg.com/babel-plugin-syntax-class-constructor-call/-/babel-plugin-syntax-class-constructor-call-6.18.0.tgz#9cb9d39fe43c8600bec8146456ddcbd4e1a76416" + +babel-plugin-syntax-class-properties@^6.8.0: + version "6.13.0" + resolved "https://registry.yarnpkg.com/babel-plugin-syntax-class-properties/-/babel-plugin-syntax-class-properties-6.13.0.tgz#d7eb23b79a317f8543962c505b827c7d6cac27de" + +babel-plugin-syntax-decorators@^6.1.18, babel-plugin-syntax-decorators@^6.13.0: + version "6.13.0" + resolved "https://registry.yarnpkg.com/babel-plugin-syntax-decorators/-/babel-plugin-syntax-decorators-6.13.0.tgz#312563b4dbde3cc806cee3e416cceeaddd11ac0b" + +babel-plugin-syntax-do-expressions@^6.8.0: + version "6.13.0" + resolved "https://registry.yarnpkg.com/babel-plugin-syntax-do-expressions/-/babel-plugin-syntax-do-expressions-6.13.0.tgz#5747756139aa26d390d09410b03744ba07e4796d" + +babel-plugin-syntax-dynamic-import@^6.18.0: + version "6.18.0" + resolved "https://registry.yarnpkg.com/babel-plugin-syntax-dynamic-import/-/babel-plugin-syntax-dynamic-import-6.18.0.tgz#8d6a26229c83745a9982a441051572caa179b1da" + +babel-plugin-syntax-exponentiation-operator@^6.8.0: + version "6.13.0" + resolved "https://registry.yarnpkg.com/babel-plugin-syntax-exponentiation-operator/-/babel-plugin-syntax-exponentiation-operator-6.13.0.tgz#9ee7e8337290da95288201a6a57f4170317830de" + +babel-plugin-syntax-export-extensions@^6.8.0: + version "6.13.0" + resolved "https://registry.yarnpkg.com/babel-plugin-syntax-export-extensions/-/babel-plugin-syntax-export-extensions-6.13.0.tgz#70a1484f0f9089a4e84ad44bac353c95b9b12721" + +babel-plugin-syntax-flow@^6.18.0: + version "6.18.0" + resolved "https://registry.yarnpkg.com/babel-plugin-syntax-flow/-/babel-plugin-syntax-flow-6.18.0.tgz#4c3ab20a2af26aa20cd25995c398c4eb70310c8d" + +babel-plugin-syntax-function-bind@^6.8.0: + version "6.13.0" + resolved "https://registry.yarnpkg.com/babel-plugin-syntax-function-bind/-/babel-plugin-syntax-function-bind-6.13.0.tgz#48c495f177bdf31a981e732f55adc0bdd2601f46" + +babel-plugin-syntax-jsx@^6.3.13, babel-plugin-syntax-jsx@^6.8.0: + version "6.18.0" + resolved "https://registry.yarnpkg.com/babel-plugin-syntax-jsx/-/babel-plugin-syntax-jsx-6.18.0.tgz#0af32a9a6e13ca7a3fd5069e62d7b0f58d0d8946" + +babel-plugin-syntax-object-rest-spread@^6.8.0: + version "6.13.0" + resolved "https://registry.yarnpkg.com/babel-plugin-syntax-object-rest-spread/-/babel-plugin-syntax-object-rest-spread-6.13.0.tgz#fd6536f2bce13836ffa3a5458c4903a597bb3bf5" + +babel-plugin-syntax-trailing-function-commas@^6.22.0: + version "6.22.0" + resolved "https://registry.yarnpkg.com/babel-plugin-syntax-trailing-function-commas/-/babel-plugin-syntax-trailing-function-commas-6.22.0.tgz#ba0360937f8d06e40180a43fe0d5616fff532cf3" + +babel-plugin-transform-async-generator-functions@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-async-generator-functions/-/babel-plugin-transform-async-generator-functions-6.24.1.tgz#f058900145fd3e9907a6ddf28da59f215258a5db" + dependencies: + babel-helper-remap-async-to-generator "^6.24.1" + babel-plugin-syntax-async-generators "^6.5.0" + babel-runtime "^6.22.0" + +babel-plugin-transform-async-to-generator@^6.22.0, babel-plugin-transform-async-to-generator@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-async-to-generator/-/babel-plugin-transform-async-to-generator-6.24.1.tgz#6536e378aff6cb1d5517ac0e40eb3e9fc8d08761" + dependencies: + babel-helper-remap-async-to-generator "^6.24.1" + babel-plugin-syntax-async-functions "^6.8.0" + babel-runtime "^6.22.0" + +babel-plugin-transform-class-constructor-call@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-class-constructor-call/-/babel-plugin-transform-class-constructor-call-6.24.1.tgz#80dc285505ac067dcb8d6c65e2f6f11ab7765ef9" + dependencies: + babel-plugin-syntax-class-constructor-call "^6.18.0" + babel-runtime "^6.22.0" + babel-template "^6.24.1" + +babel-plugin-transform-class-properties@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-class-properties/-/babel-plugin-transform-class-properties-6.24.1.tgz#6a79763ea61d33d36f37b611aa9def81a81b46ac" + dependencies: + babel-helper-function-name "^6.24.1" + babel-plugin-syntax-class-properties "^6.8.0" + babel-runtime "^6.22.0" + babel-template "^6.24.1" + +babel-plugin-transform-decorators-legacy@^1.3.4: + version "1.3.4" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-decorators-legacy/-/babel-plugin-transform-decorators-legacy-1.3.4.tgz#741b58f6c5bce9e6027e0882d9c994f04f366925" + dependencies: + babel-plugin-syntax-decorators "^6.1.18" + babel-runtime "^6.2.0" + babel-template "^6.3.0" + +babel-plugin-transform-decorators@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-decorators/-/babel-plugin-transform-decorators-6.24.1.tgz#788013d8f8c6b5222bdf7b344390dfd77569e24d" + dependencies: + babel-helper-explode-class "^6.24.1" + babel-plugin-syntax-decorators "^6.13.0" + babel-runtime "^6.22.0" + babel-template "^6.24.1" + babel-types "^6.24.1" + +babel-plugin-transform-do-expressions@^6.22.0: + version "6.22.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-do-expressions/-/babel-plugin-transform-do-expressions-6.22.0.tgz#28ccaf92812d949c2cd1281f690c8fdc468ae9bb" + dependencies: + babel-plugin-syntax-do-expressions "^6.8.0" + babel-runtime "^6.22.0" + +babel-plugin-transform-es2015-arrow-functions@^6.22.0: + version "6.22.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-arrow-functions/-/babel-plugin-transform-es2015-arrow-functions-6.22.0.tgz#452692cb711d5f79dc7f85e440ce41b9f244d221" + dependencies: + babel-runtime "^6.22.0" + +babel-plugin-transform-es2015-block-scoped-functions@^6.22.0: + version "6.22.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-block-scoped-functions/-/babel-plugin-transform-es2015-block-scoped-functions-6.22.0.tgz#bbc51b49f964d70cb8d8e0b94e820246ce3a6141" + dependencies: + babel-runtime "^6.22.0" + +babel-plugin-transform-es2015-block-scoping@^6.23.0, babel-plugin-transform-es2015-block-scoping@^6.24.1: + version "6.26.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-block-scoping/-/babel-plugin-transform-es2015-block-scoping-6.26.0.tgz#d70f5299c1308d05c12f463813b0a09e73b1895f" + dependencies: + babel-runtime "^6.26.0" + babel-template "^6.26.0" + babel-traverse "^6.26.0" + babel-types "^6.26.0" + lodash "^4.17.4" + +babel-plugin-transform-es2015-classes@^6.23.0, babel-plugin-transform-es2015-classes@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-classes/-/babel-plugin-transform-es2015-classes-6.24.1.tgz#5a4c58a50c9c9461e564b4b2a3bfabc97a2584db" + dependencies: + babel-helper-define-map "^6.24.1" + babel-helper-function-name "^6.24.1" + babel-helper-optimise-call-expression "^6.24.1" + babel-helper-replace-supers "^6.24.1" + babel-messages "^6.23.0" + babel-runtime "^6.22.0" + babel-template "^6.24.1" + babel-traverse "^6.24.1" + babel-types "^6.24.1" + +babel-plugin-transform-es2015-computed-properties@^6.22.0, babel-plugin-transform-es2015-computed-properties@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-computed-properties/-/babel-plugin-transform-es2015-computed-properties-6.24.1.tgz#6fe2a8d16895d5634f4cd999b6d3480a308159b3" + dependencies: + babel-runtime "^6.22.0" + babel-template "^6.24.1" + +babel-plugin-transform-es2015-destructuring@^6.22.0, babel-plugin-transform-es2015-destructuring@^6.23.0: + version "6.23.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-destructuring/-/babel-plugin-transform-es2015-destructuring-6.23.0.tgz#997bb1f1ab967f682d2b0876fe358d60e765c56d" + dependencies: + babel-runtime "^6.22.0" + +babel-plugin-transform-es2015-duplicate-keys@^6.22.0, babel-plugin-transform-es2015-duplicate-keys@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-duplicate-keys/-/babel-plugin-transform-es2015-duplicate-keys-6.24.1.tgz#73eb3d310ca969e3ef9ec91c53741a6f1576423e" + dependencies: + babel-runtime "^6.22.0" + babel-types "^6.24.1" + +babel-plugin-transform-es2015-for-of@^6.22.0, babel-plugin-transform-es2015-for-of@^6.23.0: + version "6.23.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-for-of/-/babel-plugin-transform-es2015-for-of-6.23.0.tgz#f47c95b2b613df1d3ecc2fdb7573623c75248691" + dependencies: + babel-runtime "^6.22.0" + +babel-plugin-transform-es2015-function-name@^6.22.0, babel-plugin-transform-es2015-function-name@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-function-name/-/babel-plugin-transform-es2015-function-name-6.24.1.tgz#834c89853bc36b1af0f3a4c5dbaa94fd8eacaa8b" + dependencies: + babel-helper-function-name "^6.24.1" + babel-runtime "^6.22.0" + babel-types "^6.24.1" + +babel-plugin-transform-es2015-literals@^6.22.0: + version "6.22.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-literals/-/babel-plugin-transform-es2015-literals-6.22.0.tgz#4f54a02d6cd66cf915280019a31d31925377ca2e" + dependencies: + babel-runtime "^6.22.0" + +babel-plugin-transform-es2015-modules-amd@^6.22.0, babel-plugin-transform-es2015-modules-amd@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-modules-amd/-/babel-plugin-transform-es2015-modules-amd-6.24.1.tgz#3b3e54017239842d6d19c3011c4bd2f00a00d154" + dependencies: + babel-plugin-transform-es2015-modules-commonjs "^6.24.1" + babel-runtime "^6.22.0" + babel-template "^6.24.1" + +babel-plugin-transform-es2015-modules-commonjs@^6.23.0, babel-plugin-transform-es2015-modules-commonjs@^6.24.1: + version "6.26.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-modules-commonjs/-/babel-plugin-transform-es2015-modules-commonjs-6.26.0.tgz#0d8394029b7dc6abe1a97ef181e00758dd2e5d8a" + dependencies: + babel-plugin-transform-strict-mode "^6.24.1" + babel-runtime "^6.26.0" + babel-template "^6.26.0" + babel-types "^6.26.0" + +babel-plugin-transform-es2015-modules-systemjs@^6.23.0, babel-plugin-transform-es2015-modules-systemjs@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-modules-systemjs/-/babel-plugin-transform-es2015-modules-systemjs-6.24.1.tgz#ff89a142b9119a906195f5f106ecf305d9407d23" + dependencies: + babel-helper-hoist-variables "^6.24.1" + babel-runtime "^6.22.0" + babel-template "^6.24.1" + +babel-plugin-transform-es2015-modules-umd@^6.23.0, babel-plugin-transform-es2015-modules-umd@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-modules-umd/-/babel-plugin-transform-es2015-modules-umd-6.24.1.tgz#ac997e6285cd18ed6176adb607d602344ad38468" + dependencies: + babel-plugin-transform-es2015-modules-amd "^6.24.1" + babel-runtime "^6.22.0" + babel-template "^6.24.1" + +babel-plugin-transform-es2015-object-super@^6.22.0, babel-plugin-transform-es2015-object-super@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-object-super/-/babel-plugin-transform-es2015-object-super-6.24.1.tgz#24cef69ae21cb83a7f8603dad021f572eb278f8d" + dependencies: + babel-helper-replace-supers "^6.24.1" + babel-runtime "^6.22.0" + +babel-plugin-transform-es2015-parameters@^6.23.0, babel-plugin-transform-es2015-parameters@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-parameters/-/babel-plugin-transform-es2015-parameters-6.24.1.tgz#57ac351ab49caf14a97cd13b09f66fdf0a625f2b" + dependencies: + babel-helper-call-delegate "^6.24.1" + babel-helper-get-function-arity "^6.24.1" + babel-runtime "^6.22.0" + babel-template "^6.24.1" + babel-traverse "^6.24.1" + babel-types "^6.24.1" + +babel-plugin-transform-es2015-shorthand-properties@^6.22.0, babel-plugin-transform-es2015-shorthand-properties@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-shorthand-properties/-/babel-plugin-transform-es2015-shorthand-properties-6.24.1.tgz#24f875d6721c87661bbd99a4622e51f14de38aa0" + dependencies: + babel-runtime "^6.22.0" + babel-types "^6.24.1" + +babel-plugin-transform-es2015-spread@^6.22.0: + version "6.22.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-spread/-/babel-plugin-transform-es2015-spread-6.22.0.tgz#d6d68a99f89aedc4536c81a542e8dd9f1746f8d1" + dependencies: + babel-runtime "^6.22.0" + +babel-plugin-transform-es2015-sticky-regex@^6.22.0, babel-plugin-transform-es2015-sticky-regex@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-sticky-regex/-/babel-plugin-transform-es2015-sticky-regex-6.24.1.tgz#00c1cdb1aca71112cdf0cf6126c2ed6b457ccdbc" + dependencies: + babel-helper-regex "^6.24.1" + babel-runtime "^6.22.0" + babel-types "^6.24.1" + +babel-plugin-transform-es2015-template-literals@^6.22.0: + version "6.22.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-template-literals/-/babel-plugin-transform-es2015-template-literals-6.22.0.tgz#a84b3450f7e9f8f1f6839d6d687da84bb1236d8d" + dependencies: + babel-runtime "^6.22.0" + +babel-plugin-transform-es2015-typeof-symbol@^6.22.0, babel-plugin-transform-es2015-typeof-symbol@^6.23.0: + version "6.23.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-typeof-symbol/-/babel-plugin-transform-es2015-typeof-symbol-6.23.0.tgz#dec09f1cddff94b52ac73d505c84df59dcceb372" + dependencies: + babel-runtime "^6.22.0" + +babel-plugin-transform-es2015-unicode-regex@^6.22.0, babel-plugin-transform-es2015-unicode-regex@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-unicode-regex/-/babel-plugin-transform-es2015-unicode-regex-6.24.1.tgz#d38b12f42ea7323f729387f18a7c5ae1faeb35e9" + dependencies: + babel-helper-regex "^6.24.1" + babel-runtime "^6.22.0" + regexpu-core "^2.0.0" + +babel-plugin-transform-es3-member-expression-literals@^6.22.0: + version "6.22.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es3-member-expression-literals/-/babel-plugin-transform-es3-member-expression-literals-6.22.0.tgz#733d3444f3ecc41bef8ed1a6a4e09657b8969ebb" + dependencies: + babel-runtime "^6.22.0" + +babel-plugin-transform-es3-property-literals@^6.22.0: + version "6.22.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-es3-property-literals/-/babel-plugin-transform-es3-property-literals-6.22.0.tgz#b2078d5842e22abf40f73e8cde9cd3711abd5758" + dependencies: + babel-runtime "^6.22.0" + +babel-plugin-transform-exponentiation-operator@^6.22.0, babel-plugin-transform-exponentiation-operator@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-exponentiation-operator/-/babel-plugin-transform-exponentiation-operator-6.24.1.tgz#2ab0c9c7f3098fa48907772bb813fe41e8de3a0e" + dependencies: + babel-helper-builder-binary-assignment-operator-visitor "^6.24.1" + babel-plugin-syntax-exponentiation-operator "^6.8.0" + babel-runtime "^6.22.0" + +babel-plugin-transform-export-extensions@^6.22.0: + version "6.22.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-export-extensions/-/babel-plugin-transform-export-extensions-6.22.0.tgz#53738b47e75e8218589eea946cbbd39109bbe653" + dependencies: + babel-plugin-syntax-export-extensions "^6.8.0" + babel-runtime "^6.22.0" + +babel-plugin-transform-flow-strip-types@^6.22.0: + version "6.22.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-flow-strip-types/-/babel-plugin-transform-flow-strip-types-6.22.0.tgz#84cb672935d43714fdc32bce84568d87441cf7cf" + dependencies: + babel-plugin-syntax-flow "^6.18.0" + babel-runtime "^6.22.0" + +babel-plugin-transform-function-bind@^6.22.0: + version "6.22.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-function-bind/-/babel-plugin-transform-function-bind-6.22.0.tgz#c6fb8e96ac296a310b8cf8ea401462407ddf6a97" + dependencies: + babel-plugin-syntax-function-bind "^6.8.0" + babel-runtime "^6.22.0" + +babel-plugin-transform-object-rest-spread@^6.22.0: + version "6.26.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-object-rest-spread/-/babel-plugin-transform-object-rest-spread-6.26.0.tgz#0f36692d50fef6b7e2d4b3ac1478137a963b7b06" + dependencies: + babel-plugin-syntax-object-rest-spread "^6.8.0" + babel-runtime "^6.26.0" + +babel-plugin-transform-react-display-name@^6.23.0: + version "6.25.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-react-display-name/-/babel-plugin-transform-react-display-name-6.25.0.tgz#67e2bf1f1e9c93ab08db96792e05392bf2cc28d1" + dependencies: + babel-runtime "^6.22.0" + +babel-plugin-transform-react-jsx-self@^6.22.0: + version "6.22.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-react-jsx-self/-/babel-plugin-transform-react-jsx-self-6.22.0.tgz#df6d80a9da2612a121e6ddd7558bcbecf06e636e" + dependencies: + babel-plugin-syntax-jsx "^6.8.0" + babel-runtime "^6.22.0" + +babel-plugin-transform-react-jsx-source@^6.22.0: + version "6.22.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-react-jsx-source/-/babel-plugin-transform-react-jsx-source-6.22.0.tgz#66ac12153f5cd2d17b3c19268f4bf0197f44ecd6" + dependencies: + babel-plugin-syntax-jsx "^6.8.0" + babel-runtime "^6.22.0" + +babel-plugin-transform-react-jsx@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-react-jsx/-/babel-plugin-transform-react-jsx-6.24.1.tgz#840a028e7df460dfc3a2d29f0c0d91f6376e66a3" + dependencies: + babel-helper-builder-react-jsx "^6.24.1" + babel-plugin-syntax-jsx "^6.8.0" + babel-runtime "^6.22.0" + +babel-plugin-transform-regenerator@^6.22.0, babel-plugin-transform-regenerator@^6.24.1: + version "6.26.0" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-regenerator/-/babel-plugin-transform-regenerator-6.26.0.tgz#e0703696fbde27f0a3efcacf8b4dca2f7b3a8f2f" + dependencies: + regenerator-transform "^0.10.0" + +babel-plugin-transform-strict-mode@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-plugin-transform-strict-mode/-/babel-plugin-transform-strict-mode-6.24.1.tgz#d5faf7aa578a65bbe591cf5edae04a0c67020758" + dependencies: + babel-runtime "^6.22.0" + babel-types "^6.24.1" + +babel-polyfill@^6.23.0: + version "6.26.0" + resolved "https://registry.yarnpkg.com/babel-polyfill/-/babel-polyfill-6.26.0.tgz#379937abc67d7895970adc621f284cd966cf2153" + dependencies: + babel-runtime "^6.26.0" + core-js "^2.5.0" + regenerator-runtime "^0.10.5" + +babel-preset-env@^1.6.0: + version "1.6.0" + resolved "https://registry.yarnpkg.com/babel-preset-env/-/babel-preset-env-1.6.0.tgz#2de1c782a780a0a5d605d199c957596da43c44e4" + dependencies: + babel-plugin-check-es2015-constants "^6.22.0" + babel-plugin-syntax-trailing-function-commas "^6.22.0" + babel-plugin-transform-async-to-generator "^6.22.0" + babel-plugin-transform-es2015-arrow-functions "^6.22.0" + babel-plugin-transform-es2015-block-scoped-functions "^6.22.0" + babel-plugin-transform-es2015-block-scoping "^6.23.0" + babel-plugin-transform-es2015-classes "^6.23.0" + babel-plugin-transform-es2015-computed-properties "^6.22.0" + babel-plugin-transform-es2015-destructuring "^6.23.0" + babel-plugin-transform-es2015-duplicate-keys "^6.22.0" + babel-plugin-transform-es2015-for-of "^6.23.0" + babel-plugin-transform-es2015-function-name "^6.22.0" + babel-plugin-transform-es2015-literals "^6.22.0" + babel-plugin-transform-es2015-modules-amd "^6.22.0" + babel-plugin-transform-es2015-modules-commonjs "^6.23.0" + babel-plugin-transform-es2015-modules-systemjs "^6.23.0" + babel-plugin-transform-es2015-modules-umd "^6.23.0" + babel-plugin-transform-es2015-object-super "^6.22.0" + babel-plugin-transform-es2015-parameters "^6.23.0" + babel-plugin-transform-es2015-shorthand-properties "^6.22.0" + babel-plugin-transform-es2015-spread "^6.22.0" + babel-plugin-transform-es2015-sticky-regex "^6.22.0" + babel-plugin-transform-es2015-template-literals "^6.22.0" + babel-plugin-transform-es2015-typeof-symbol "^6.23.0" + babel-plugin-transform-es2015-unicode-regex "^6.22.0" + babel-plugin-transform-exponentiation-operator "^6.22.0" + babel-plugin-transform-regenerator "^6.22.0" + browserslist "^2.1.2" + invariant "^2.2.2" + semver "^5.3.0" + +babel-preset-es2015@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-preset-es2015/-/babel-preset-es2015-6.24.1.tgz#d44050d6bc2c9feea702aaf38d727a0210538939" + dependencies: + babel-plugin-check-es2015-constants "^6.22.0" + babel-plugin-transform-es2015-arrow-functions "^6.22.0" + babel-plugin-transform-es2015-block-scoped-functions "^6.22.0" + babel-plugin-transform-es2015-block-scoping "^6.24.1" + babel-plugin-transform-es2015-classes "^6.24.1" + babel-plugin-transform-es2015-computed-properties "^6.24.1" + babel-plugin-transform-es2015-destructuring "^6.22.0" + babel-plugin-transform-es2015-duplicate-keys "^6.24.1" + babel-plugin-transform-es2015-for-of "^6.22.0" + babel-plugin-transform-es2015-function-name "^6.24.1" + babel-plugin-transform-es2015-literals "^6.22.0" + babel-plugin-transform-es2015-modules-amd "^6.24.1" + babel-plugin-transform-es2015-modules-commonjs "^6.24.1" + babel-plugin-transform-es2015-modules-systemjs "^6.24.1" + babel-plugin-transform-es2015-modules-umd "^6.24.1" + babel-plugin-transform-es2015-object-super "^6.24.1" + babel-plugin-transform-es2015-parameters "^6.24.1" + babel-plugin-transform-es2015-shorthand-properties "^6.24.1" + babel-plugin-transform-es2015-spread "^6.22.0" + babel-plugin-transform-es2015-sticky-regex "^6.24.1" + babel-plugin-transform-es2015-template-literals "^6.22.0" + babel-plugin-transform-es2015-typeof-symbol "^6.22.0" + babel-plugin-transform-es2015-unicode-regex "^6.24.1" + babel-plugin-transform-regenerator "^6.24.1" + +babel-preset-flow@^6.23.0: + version "6.23.0" + resolved "https://registry.yarnpkg.com/babel-preset-flow/-/babel-preset-flow-6.23.0.tgz#e71218887085ae9a24b5be4169affb599816c49d" + dependencies: + babel-plugin-transform-flow-strip-types "^6.22.0" + +babel-preset-jest@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/babel-preset-jest/-/babel-preset-jest-21.0.2.tgz#9db25def2329f49eace3f5ea0de42a0b898d12cc" + dependencies: + babel-plugin-jest-hoist "^21.0.2" + +babel-preset-react@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-preset-react/-/babel-preset-react-6.24.1.tgz#ba69dfaea45fc3ec639b6a4ecea6e17702c91380" + dependencies: + babel-plugin-syntax-jsx "^6.3.13" + babel-plugin-transform-react-display-name "^6.23.0" + babel-plugin-transform-react-jsx "^6.24.1" + babel-plugin-transform-react-jsx-self "^6.22.0" + babel-plugin-transform-react-jsx-source "^6.22.0" + babel-preset-flow "^6.23.0" + +babel-preset-stage-0@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-preset-stage-0/-/babel-preset-stage-0-6.24.1.tgz#5642d15042f91384d7e5af8bc88b1db95b039e6a" + dependencies: + babel-plugin-transform-do-expressions "^6.22.0" + babel-plugin-transform-function-bind "^6.22.0" + babel-preset-stage-1 "^6.24.1" + +babel-preset-stage-1@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-preset-stage-1/-/babel-preset-stage-1-6.24.1.tgz#7692cd7dcd6849907e6ae4a0a85589cfb9e2bfb0" + dependencies: + babel-plugin-transform-class-constructor-call "^6.24.1" + babel-plugin-transform-export-extensions "^6.22.0" + babel-preset-stage-2 "^6.24.1" + +babel-preset-stage-2@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-preset-stage-2/-/babel-preset-stage-2-6.24.1.tgz#d9e2960fb3d71187f0e64eec62bc07767219bdc1" + dependencies: + babel-plugin-syntax-dynamic-import "^6.18.0" + babel-plugin-transform-class-properties "^6.24.1" + babel-plugin-transform-decorators "^6.24.1" + babel-preset-stage-3 "^6.24.1" + +babel-preset-stage-3@^6.24.1: + version "6.24.1" + resolved "https://registry.yarnpkg.com/babel-preset-stage-3/-/babel-preset-stage-3-6.24.1.tgz#836ada0a9e7a7fa37cb138fb9326f87934a48395" + dependencies: + babel-plugin-syntax-trailing-function-commas "^6.22.0" + babel-plugin-transform-async-generator-functions "^6.24.1" + babel-plugin-transform-async-to-generator "^6.24.1" + babel-plugin-transform-exponentiation-operator "^6.24.1" + babel-plugin-transform-object-rest-spread "^6.22.0" + +babel-register@^6.26.0: + version "6.26.0" + resolved "https://registry.yarnpkg.com/babel-register/-/babel-register-6.26.0.tgz#6ed021173e2fcb486d7acb45c6009a856f647071" + dependencies: + babel-core "^6.26.0" + babel-runtime "^6.26.0" + core-js "^2.5.0" + home-or-tmp "^2.0.0" + lodash "^4.17.4" + mkdirp "^0.5.1" + source-map-support "^0.4.15" + +babel-runtime@^6.18.0, babel-runtime@^6.2.0, babel-runtime@^6.22.0, babel-runtime@^6.26.0: + version "6.26.0" + resolved "https://registry.yarnpkg.com/babel-runtime/-/babel-runtime-6.26.0.tgz#965c7058668e82b55d7bfe04ff2337bc8b5647fe" + dependencies: + core-js "^2.4.0" + regenerator-runtime "^0.11.0" + +babel-template@^6.16.0, babel-template@^6.24.1, babel-template@^6.26.0, babel-template@^6.3.0: + version "6.26.0" + resolved "https://registry.yarnpkg.com/babel-template/-/babel-template-6.26.0.tgz#de03e2d16396b069f46dd9fff8521fb1a0e35e02" + dependencies: + babel-runtime "^6.26.0" + babel-traverse "^6.26.0" + babel-types "^6.26.0" + babylon "^6.18.0" + lodash "^4.17.4" + +babel-traverse@^6.18.0, babel-traverse@^6.23.1, babel-traverse@^6.24.1, babel-traverse@^6.26.0: + version "6.26.0" + resolved "https://registry.yarnpkg.com/babel-traverse/-/babel-traverse-6.26.0.tgz#46a9cbd7edcc62c8e5c064e2d2d8d0f4035766ee" + dependencies: + babel-code-frame "^6.26.0" + babel-messages "^6.23.0" + babel-runtime "^6.26.0" + babel-types "^6.26.0" + babylon "^6.18.0" + debug "^2.6.8" + globals "^9.18.0" + invariant "^2.2.2" + lodash "^4.17.4" + +babel-types@^6.18.0, babel-types@^6.19.0, babel-types@^6.23.0, babel-types@^6.24.1, babel-types@^6.26.0: + version "6.26.0" + resolved "https://registry.yarnpkg.com/babel-types/-/babel-types-6.26.0.tgz#a3b073f94ab49eb6fa55cd65227a334380632497" + dependencies: + babel-runtime "^6.26.0" + esutils "^2.0.2" + lodash "^4.17.4" + to-fast-properties "^1.0.3" + +babylon@^6.17.0, babylon@^6.18.0: + version "6.18.0" + resolved "https://registry.yarnpkg.com/babylon/-/babylon-6.18.0.tgz#af2f3b88fa6f5c1e4c634d1a0f8eac4f55b395e3" + +babylon@~7.0.0-beta.19: + version "7.0.0-beta.22" + resolved "https://registry.yarnpkg.com/babylon/-/babylon-7.0.0-beta.22.tgz#74f0ad82ed7c7c3cfeab74cf684f815104161b65" + +balanced-match@^0.4.2: + version "0.4.2" + resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-0.4.2.tgz#cb3f3e3c732dc0f01ee70b403f302e61d7709838" + +balanced-match@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.0.tgz#89b4d199ab2bee49de164ea02b89ce462d71b767" + +baobab-react@^2.1.2: + version "2.1.2" + resolved "https://registry.yarnpkg.com/baobab-react/-/baobab-react-2.1.2.tgz#76bd9d48e4befb52ac4fc951be8cce793c5b9ed2" + +baobab@^2.4.3: + version "2.4.3" + resolved "https://registry.yarnpkg.com/baobab/-/baobab-2.4.3.tgz#9655d6f1a69ce66a82fc6391773ca6a9f7a25a9d" + dependencies: + emmett "^3.1.1" + +base64-js@^1.0.2: + version "1.2.1" + resolved "https://registry.yarnpkg.com/base64-js/-/base64-js-1.2.1.tgz#a91947da1f4a516ea38e5b4ec0ec3773675e0886" + +batch@0.6.1: + version "0.6.1" + resolved "https://registry.yarnpkg.com/batch/-/batch-0.6.1.tgz#dc34314f4e679318093fc760272525f94bf25c16" + +bcrypt-pbkdf@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.1.tgz#63bc5dcb61331b92bc05fd528953c33462a06f8d" + dependencies: + tweetnacl "^0.14.3" + +big.js@^3.1.3: + version "3.1.3" + resolved "https://registry.yarnpkg.com/big.js/-/big.js-3.1.3.tgz#4cada2193652eb3ca9ec8e55c9015669c9806978" + +binary-extensions@^1.0.0: + version "1.10.0" + resolved "https://registry.yarnpkg.com/binary-extensions/-/binary-extensions-1.10.0.tgz#9aeb9a6c5e88638aad171e167f5900abe24835d0" + +block-stream@*: + version "0.0.9" + resolved "https://registry.yarnpkg.com/block-stream/-/block-stream-0.0.9.tgz#13ebfe778a03205cfe03751481ebb4b3300c126a" + dependencies: + inherits "~2.0.0" + +bluebird@^3.4.7, bluebird@~3.5.0: + version "3.5.0" + resolved "https://registry.yarnpkg.com/bluebird/-/bluebird-3.5.0.tgz#791420d7f551eea2897453a8a77653f96606d67c" + +bn.js@^4.0.0, bn.js@^4.1.0, bn.js@^4.1.1, bn.js@^4.4.0: + version "4.11.8" + resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-4.11.8.tgz#2cde09eb5ee341f484746bb0309b3253b1b1442f" + +bonjour@^3.5.0: + version "3.5.0" + resolved "https://registry.yarnpkg.com/bonjour/-/bonjour-3.5.0.tgz#8e890a183d8ee9a2393b3844c691a42bcf7bc9f5" + dependencies: + array-flatten "^2.1.0" + deep-equal "^1.0.1" + dns-equal "^1.0.0" + dns-txt "^2.0.2" + multicast-dns "^6.0.1" + multicast-dns-service-types "^1.1.0" + +boolbase@~1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/boolbase/-/boolbase-1.0.0.tgz#68dff5fbe60c51eb37725ea9e3ed310dcc1e776e" + +boom@2.x.x: + version "2.10.1" + resolved "https://registry.yarnpkg.com/boom/-/boom-2.10.1.tgz#39c8918ceff5799f83f9492a848f625add0c766f" + dependencies: + hoek "2.x.x" + +brace-expansion@^1.1.7: + version "1.1.8" + resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.8.tgz#c07b211c7c952ec1f8efd51a77ef0d1d3990a292" + dependencies: + balanced-match "^1.0.0" + concat-map "0.0.1" + +braces@^1.8.2: + version "1.8.5" + resolved "https://registry.yarnpkg.com/braces/-/braces-1.8.5.tgz#ba77962e12dff969d6b76711e914b737857bf6a7" + dependencies: + expand-range "^1.8.1" + preserve "^0.2.0" + repeat-element "^1.1.2" + +brorand@^1.0.1: + version "1.1.0" + resolved "https://registry.yarnpkg.com/brorand/-/brorand-1.1.0.tgz#12c25efe40a45e3c323eb8675a0a0ce57b22371f" + +browser-resolve@^1.11.2: + version "1.11.2" + resolved "https://registry.yarnpkg.com/browser-resolve/-/browser-resolve-1.11.2.tgz#8ff09b0a2c421718a1051c260b32e48f442938ce" + dependencies: + resolve "1.1.7" + +browserify-aes@^1.0.0, browserify-aes@^1.0.4: + version "1.0.8" + resolved "https://registry.yarnpkg.com/browserify-aes/-/browserify-aes-1.0.8.tgz#c8fa3b1b7585bb7ba77c5560b60996ddec6d5309" + dependencies: + buffer-xor "^1.0.3" + cipher-base "^1.0.0" + create-hash "^1.1.0" + evp_bytestokey "^1.0.3" + inherits "^2.0.1" + safe-buffer "^5.0.1" + +browserify-cipher@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/browserify-cipher/-/browserify-cipher-1.0.0.tgz#9988244874bf5ed4e28da95666dcd66ac8fc363a" + dependencies: + browserify-aes "^1.0.4" + browserify-des "^1.0.0" + evp_bytestokey "^1.0.0" + +browserify-des@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/browserify-des/-/browserify-des-1.0.0.tgz#daa277717470922ed2fe18594118a175439721dd" + dependencies: + cipher-base "^1.0.1" + des.js "^1.0.0" + inherits "^2.0.1" + +browserify-rsa@^4.0.0: + version "4.0.1" + resolved "https://registry.yarnpkg.com/browserify-rsa/-/browserify-rsa-4.0.1.tgz#21e0abfaf6f2029cf2fafb133567a701d4135524" + dependencies: + bn.js "^4.1.0" + randombytes "^2.0.1" + +browserify-sign@^4.0.0: + version "4.0.4" + resolved "https://registry.yarnpkg.com/browserify-sign/-/browserify-sign-4.0.4.tgz#aa4eb68e5d7b658baa6bf6a57e630cbd7a93d298" + dependencies: + bn.js "^4.1.1" + browserify-rsa "^4.0.0" + create-hash "^1.1.0" + create-hmac "^1.1.2" + elliptic "^6.0.0" + inherits "^2.0.1" + parse-asn1 "^5.0.0" + +browserify-zlib@^0.1.4: + version "0.1.4" + resolved "https://registry.yarnpkg.com/browserify-zlib/-/browserify-zlib-0.1.4.tgz#bb35f8a519f600e0fa6b8485241c979d0141fb2d" + dependencies: + pako "~0.2.0" + +browserslist@^1.3.6, browserslist@^1.5.2, browserslist@^1.7.6: + version "1.7.7" + resolved "https://registry.yarnpkg.com/browserslist/-/browserslist-1.7.7.tgz#0bd76704258be829b2398bb50e4b62d1a166b0b9" + dependencies: + caniuse-db "^1.0.30000639" + electron-to-chromium "^1.2.7" + +browserslist@^2.1.2: + version "2.4.0" + resolved "https://registry.yarnpkg.com/browserslist/-/browserslist-2.4.0.tgz#693ee93d01e66468a6348da5498e011f578f87f8" + dependencies: + caniuse-lite "^1.0.30000718" + electron-to-chromium "^1.3.18" + +bser@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/bser/-/bser-2.0.0.tgz#9ac78d3ed5d915804fd87acb158bc797147a1719" + dependencies: + node-int64 "^0.4.0" + +buffer-indexof@^1.0.0: + version "1.1.1" + resolved "https://registry.yarnpkg.com/buffer-indexof/-/buffer-indexof-1.1.1.tgz#52fabcc6a606d1a00302802648ef68f639da268c" + +buffer-xor@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/buffer-xor/-/buffer-xor-1.0.3.tgz#26e61ed1422fb70dd42e6e36729ed51d855fe8d9" + +buffer@^4.3.0: + version "4.9.1" + resolved "https://registry.yarnpkg.com/buffer/-/buffer-4.9.1.tgz#6d1bb601b07a4efced97094132093027c95bc298" + dependencies: + base64-js "^1.0.2" + ieee754 "^1.1.4" + isarray "^1.0.0" + +builtin-modules@^1.0.0, builtin-modules@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/builtin-modules/-/builtin-modules-1.1.1.tgz#270f076c5a72c02f5b65a47df94c5fe3a278892f" + +builtin-status-codes@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/builtin-status-codes/-/builtin-status-codes-3.0.0.tgz#85982878e21b98e1c66425e03d0174788f569ee8" + +bytes@2.5.0: + version "2.5.0" + resolved "https://registry.yarnpkg.com/bytes/-/bytes-2.5.0.tgz#4c9423ea2d252c270c41b2bdefeff9bb6b62c06a" + +caller-path@^0.1.0: + version "0.1.0" + resolved "https://registry.yarnpkg.com/caller-path/-/caller-path-0.1.0.tgz#94085ef63581ecd3daa92444a8fe94e82577751f" + dependencies: + callsites "^0.2.0" + +callsites@^0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/callsites/-/callsites-0.2.0.tgz#afab96262910a7f33c19a5775825c69f34e350ca" + +callsites@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/callsites/-/callsites-2.0.0.tgz#06eb84f00eea413da86affefacbffb36093b3c50" + +camel-case@3.0.x: + version "3.0.0" + resolved "https://registry.yarnpkg.com/camel-case/-/camel-case-3.0.0.tgz#ca3c3688a4e9cf3a4cda777dc4dcbc713249cf73" + dependencies: + no-case "^2.2.0" + upper-case "^1.1.1" + +camelcase-keys@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/camelcase-keys/-/camelcase-keys-2.1.0.tgz#308beeaffdf28119051efa1d932213c91b8f92e7" + dependencies: + camelcase "^2.0.0" + map-obj "^1.0.0" + +camelcase@^1.0.2: + version "1.2.1" + resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-1.2.1.tgz#9bb5304d2e0b56698b2c758b08a3eaa9daa58a39" + +camelcase@^2.0.0: + version "2.1.1" + resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-2.1.1.tgz#7c1d16d679a1bbe59ca02cacecfb011e201f5a1f" + +camelcase@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-3.0.0.tgz#32fc4b9fcdaf845fcdf7e73bb97cac2261f0ab0a" + +camelcase@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-4.1.0.tgz#d545635be1e33c542649c69173e5de6acfae34dd" + +caniuse-api@^1.5.2: + version "1.6.1" + resolved "https://registry.yarnpkg.com/caniuse-api/-/caniuse-api-1.6.1.tgz#b534e7c734c4f81ec5fbe8aca2ad24354b962c6c" + dependencies: + browserslist "^1.3.6" + caniuse-db "^1.0.30000529" + lodash.memoize "^4.1.2" + lodash.uniq "^4.5.0" + +caniuse-db@^1.0.30000529, caniuse-db@^1.0.30000634, caniuse-db@^1.0.30000639: + version "1.0.30000726" + resolved "https://registry.yarnpkg.com/caniuse-db/-/caniuse-db-1.0.30000726.tgz#9bb742f8d026a62df873bc03c06843d2255b60d7" + +caniuse-lite@^1.0.30000718: + version "1.0.30000726" + resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30000726.tgz#966a753fa107a09d4131cf8b3d616723a06ccf7e" + +caseless@~0.12.0: + version "0.12.0" + resolved "https://registry.yarnpkg.com/caseless/-/caseless-0.12.0.tgz#1b681c21ff84033c826543090689420d187151dc" + +catharsis@~0.8.9: + version "0.8.9" + resolved "https://registry.yarnpkg.com/catharsis/-/catharsis-0.8.9.tgz#98cc890ca652dd2ef0e70b37925310ff9e90fc8b" + dependencies: + underscore-contrib "~0.3.0" + +center-align@^0.1.1: + version "0.1.3" + resolved "https://registry.yarnpkg.com/center-align/-/center-align-0.1.3.tgz#aa0d32629b6ee972200411cbd4461c907bc2b7ad" + dependencies: + align-text "^0.1.3" + lazy-cache "^1.0.3" + +chain-function@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/chain-function/-/chain-function-1.0.0.tgz#0d4ab37e7e18ead0bdc47b920764118ce58733dc" + +chalk@^1.1.1, chalk@^1.1.3: + version "1.1.3" + resolved "https://registry.yarnpkg.com/chalk/-/chalk-1.1.3.tgz#a8115c55e4a702fe4d150abd3872822a7e09fc98" + dependencies: + ansi-styles "^2.2.1" + escape-string-regexp "^1.0.2" + has-ansi "^2.0.0" + strip-ansi "^3.0.0" + supports-color "^2.0.0" + +chalk@^2.0.0, chalk@^2.0.1, chalk@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/chalk/-/chalk-2.1.0.tgz#ac5becf14fa21b99c6c92ca7a7d7cfd5b17e743e" + dependencies: + ansi-styles "^3.1.0" + escape-string-regexp "^1.0.5" + supports-color "^4.0.0" + +cheerio@^0.22.0: + version "0.22.0" + resolved "https://registry.yarnpkg.com/cheerio/-/cheerio-0.22.0.tgz#a9baa860a3f9b595a6b81b1a86873121ed3a269e" + dependencies: + css-select "~1.2.0" + dom-serializer "~0.1.0" + entities "~1.1.1" + htmlparser2 "^3.9.1" + lodash.assignin "^4.0.9" + lodash.bind "^4.1.4" + lodash.defaults "^4.0.1" + lodash.filter "^4.4.0" + lodash.flatten "^4.2.0" + lodash.foreach "^4.3.0" + lodash.map "^4.4.0" + lodash.merge "^4.4.0" + lodash.pick "^4.2.1" + lodash.reduce "^4.4.0" + lodash.reject "^4.4.0" + lodash.some "^4.4.0" + +chokidar@^1.6.0, chokidar@^1.7.0: + version "1.7.0" + resolved "https://registry.yarnpkg.com/chokidar/-/chokidar-1.7.0.tgz#798e689778151c8076b4b360e5edd28cda2bb468" + dependencies: + anymatch "^1.3.0" + async-each "^1.0.0" + glob-parent "^2.0.0" + inherits "^2.0.1" + is-binary-path "^1.0.0" + is-glob "^2.0.0" + path-is-absolute "^1.0.0" + readdirp "^2.0.0" + optionalDependencies: + fsevents "^1.0.0" + +ci-info@^1.0.0: + version "1.1.1" + resolved "https://registry.yarnpkg.com/ci-info/-/ci-info-1.1.1.tgz#47b44df118c48d2597b56d342e7e25791060171a" + +cipher-base@^1.0.0, cipher-base@^1.0.1, cipher-base@^1.0.3: + version "1.0.4" + resolved "https://registry.yarnpkg.com/cipher-base/-/cipher-base-1.0.4.tgz#8760e4ecc272f4c363532f926d874aae2c1397de" + dependencies: + inherits "^2.0.1" + safe-buffer "^5.0.1" + +circular-json@^0.3.1: + version "0.3.3" + resolved "https://registry.yarnpkg.com/circular-json/-/circular-json-0.3.3.tgz#815c99ea84f6809529d2f45791bdf82711352d66" + +clap@^1.0.9: + version "1.2.0" + resolved "https://registry.yarnpkg.com/clap/-/clap-1.2.0.tgz#59c90fe3e137104746ff19469a27a634ff68c857" + dependencies: + chalk "^1.1.3" + +classnames@^2.2.5: + version "2.2.5" + resolved "https://registry.yarnpkg.com/classnames/-/classnames-2.2.5.tgz#fb3801d453467649ef3603c7d61a02bd129bde6d" + +clean-css@4.1.x: + version "4.1.8" + resolved "https://registry.yarnpkg.com/clean-css/-/clean-css-4.1.8.tgz#061455b2494a750ac98f46d8d5ebb17c679ea9d1" + dependencies: + source-map "0.5.x" + +cli-cursor@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/cli-cursor/-/cli-cursor-2.1.0.tgz#b35dac376479facc3e94747d41d0d0f5238ffcb5" + dependencies: + restore-cursor "^2.0.0" + +cli-width@^2.0.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/cli-width/-/cli-width-2.2.0.tgz#ff19ede8a9a5e579324147b0c11f0fbcbabed639" + +cliui@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/cliui/-/cliui-2.1.0.tgz#4b475760ff80264c762c3a1719032e91c7fea0d1" + dependencies: + center-align "^0.1.1" + right-align "^0.1.1" + wordwrap "0.0.2" + +cliui@^3.2.0: + version "3.2.0" + resolved "https://registry.yarnpkg.com/cliui/-/cliui-3.2.0.tgz#120601537a916d29940f934da3b48d585a39213d" + dependencies: + string-width "^1.0.1" + strip-ansi "^3.0.1" + wrap-ansi "^2.0.0" + +clone@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/clone/-/clone-1.0.2.tgz#260b7a99ebb1edfe247538175f783243cb19d149" + +clone@^2.1.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/clone/-/clone-2.1.1.tgz#d217d1e961118e3ac9a4b8bba3285553bf647cdb" + +co@^4.6.0: + version "4.6.0" + resolved "https://registry.yarnpkg.com/co/-/co-4.6.0.tgz#6ea6bdf3d853ae54ccb8e47bfa0bf3f9031fb184" + +coa@~1.0.1: + version "1.0.4" + resolved "https://registry.yarnpkg.com/coa/-/coa-1.0.4.tgz#a9ef153660d6a86a8bdec0289a5c684d217432fd" + dependencies: + q "^1.1.2" + +code-point-at@^1.0.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/code-point-at/-/code-point-at-1.1.0.tgz#0d070b4d043a5bea33a2f1a40e2edb3d9a4ccf77" + +color-convert@^1.3.0, color-convert@^1.9.0: + version "1.9.0" + resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-1.9.0.tgz#1accf97dd739b983bf994d56fec8f95853641b7a" + dependencies: + color-name "^1.1.1" + +color-name@^1.0.0, color-name@^1.1.1: + version "1.1.3" + resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.3.tgz#a7d0558bd89c42f795dd42328f740831ca53bc25" + +color-string@^0.3.0: + version "0.3.0" + resolved "https://registry.yarnpkg.com/color-string/-/color-string-0.3.0.tgz#27d46fb67025c5c2fa25993bfbf579e47841b991" + dependencies: + color-name "^1.0.0" + +color@^0.11.0: + version "0.11.4" + resolved "https://registry.yarnpkg.com/color/-/color-0.11.4.tgz#6d7b5c74fb65e841cd48792ad1ed5e07b904d764" + dependencies: + clone "^1.0.2" + color-convert "^1.3.0" + color-string "^0.3.0" + +colormin@^1.0.5: + version "1.1.2" + resolved "https://registry.yarnpkg.com/colormin/-/colormin-1.1.2.tgz#ea2f7420a72b96881a38aae59ec124a6f7298133" + dependencies: + color "^0.11.0" + css-color-names "0.0.4" + has "^1.0.1" + +colors@~1.1.2: + version "1.1.2" + resolved "https://registry.yarnpkg.com/colors/-/colors-1.1.2.tgz#168a4701756b6a7f51a12ce0c97bfa28c084ed63" + +combined-stream@^1.0.5, combined-stream@~1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/combined-stream/-/combined-stream-1.0.5.tgz#938370a57b4a51dea2c77c15d5c5fdf895164009" + dependencies: + delayed-stream "~1.0.0" + +commander@2.11.x, commander@^2.8.0, commander@~2.11.0: + version "2.11.0" + resolved "https://registry.yarnpkg.com/commander/-/commander-2.11.0.tgz#157152fd1e7a6c8d98a5b715cf376df928004563" + +commander@~2.20.0: + version "2.20.0" + resolved "https://registry.yarnpkg.com/commander/-/commander-2.20.0.tgz#d58bb2b5c1ee8f87b0d340027e9e94e222c5a422" + integrity sha512-7j2y+40w61zy6YC2iRNpUe/NwhNyoXrYpHMrSunaMG64nRnaf96zO/KMQR4OyN/UnE5KLyEBnKHd4aG3rskjpQ== + +commondir@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/commondir/-/commondir-1.0.1.tgz#ddd800da0c66127393cca5950ea968a3aaf1253b" + +compressible@~2.0.10: + version "2.0.11" + resolved "https://registry.yarnpkg.com/compressible/-/compressible-2.0.11.tgz#16718a75de283ed8e604041625a2064586797d8a" + dependencies: + mime-db ">= 1.29.0 < 2" + +compression@^1.5.2: + version "1.7.0" + resolved "https://registry.yarnpkg.com/compression/-/compression-1.7.0.tgz#030c9f198f1643a057d776a738e922da4373012d" + dependencies: + accepts "~1.3.3" + bytes "2.5.0" + compressible "~2.0.10" + debug "2.6.8" + on-headers "~1.0.1" + safe-buffer "5.1.1" + vary "~1.1.1" + +concat-map@0.0.1: + version "0.0.1" + resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b" + +concat-stream@^1.6.0: + version "1.6.0" + resolved "https://registry.yarnpkg.com/concat-stream/-/concat-stream-1.6.0.tgz#0aac662fd52be78964d5532f694784e70110acf7" + dependencies: + inherits "^2.0.3" + readable-stream "^2.2.2" + typedarray "^0.0.6" + +connect-history-api-fallback@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/connect-history-api-fallback/-/connect-history-api-fallback-1.3.0.tgz#e51d17f8f0ef0db90a64fdb47de3051556e9f169" + +console-browserify@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/console-browserify/-/console-browserify-1.1.0.tgz#f0241c45730a9fc6323b206dbf38edc741d0bb10" + dependencies: + date-now "^0.1.4" + +console-control-strings@^1.0.0, console-control-strings@~1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/console-control-strings/-/console-control-strings-1.1.0.tgz#3d7cf4464db6446ea644bf4b39507f9851008e8e" + +constants-browserify@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/constants-browserify/-/constants-browserify-1.0.0.tgz#c20b96d8c617748aaf1c16021760cd27fcb8cb75" + +contains-path@^0.1.0: + version "0.1.0" + resolved "https://registry.yarnpkg.com/contains-path/-/contains-path-0.1.0.tgz#fe8cf184ff6670b6baef01a9d4861a5cbec4120a" + +content-disposition@0.5.2: + version "0.5.2" + resolved "https://registry.yarnpkg.com/content-disposition/-/content-disposition-0.5.2.tgz#0cf68bb9ddf5f2be7961c3a85178cb85dba78cb4" + +content-type-parser@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/content-type-parser/-/content-type-parser-1.0.1.tgz#c3e56988c53c65127fb46d4032a3a900246fdc94" + +content-type@~1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/content-type/-/content-type-1.0.2.tgz#b7d113aee7a8dd27bd21133c4dc2529df1721eed" + +convert-source-map@^1.4.0, convert-source-map@^1.5.0: + version "1.5.0" + resolved "https://registry.yarnpkg.com/convert-source-map/-/convert-source-map-1.5.0.tgz#9acd70851c6d5dfdd93d9282e5edf94a03ff46b5" + +cookie-signature@1.0.6: + version "1.0.6" + resolved "https://registry.yarnpkg.com/cookie-signature/-/cookie-signature-1.0.6.tgz#e303a882b342cc3ee8ca513a79999734dab3ae2c" + +cookie@0.3.1: + version "0.3.1" + resolved "https://registry.yarnpkg.com/cookie/-/cookie-0.3.1.tgz#e7e0a1f9ef43b4c8ba925c5c5a96e806d16873bb" + +core-js@^1.0.0: + version "1.2.7" + resolved "https://registry.yarnpkg.com/core-js/-/core-js-1.2.7.tgz#652294c14651db28fa93bd2d5ff2983a4f08c636" + +core-js@^2.4.0, core-js@^2.5.0: + version "2.5.1" + resolved "https://registry.yarnpkg.com/core-js/-/core-js-2.5.1.tgz#ae6874dc66937789b80754ff5428df66819ca50b" + +core-util-is@1.0.2, core-util-is@~1.0.0: + version "1.0.2" + resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.2.tgz#b5fd54220aa2bc5ab57aab7140c940754503c1a7" + +create-ecdh@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/create-ecdh/-/create-ecdh-4.0.0.tgz#888c723596cdf7612f6498233eebd7a35301737d" + dependencies: + bn.js "^4.1.0" + elliptic "^6.0.0" + +create-hash@^1.1.0, create-hash@^1.1.2: + version "1.1.3" + resolved "https://registry.yarnpkg.com/create-hash/-/create-hash-1.1.3.tgz#606042ac8b9262750f483caddab0f5819172d8fd" + dependencies: + cipher-base "^1.0.1" + inherits "^2.0.1" + ripemd160 "^2.0.0" + sha.js "^2.4.0" + +create-hmac@^1.1.0, create-hmac@^1.1.2, create-hmac@^1.1.4: + version "1.1.6" + resolved "https://registry.yarnpkg.com/create-hmac/-/create-hmac-1.1.6.tgz#acb9e221a4e17bdb076e90657c42b93e3726cf06" + dependencies: + cipher-base "^1.0.3" + create-hash "^1.1.0" + inherits "^2.0.1" + ripemd160 "^2.0.0" + safe-buffer "^5.0.1" + sha.js "^2.4.8" + +cross-env@^5.0.3: + version "5.0.5" + resolved "https://registry.yarnpkg.com/cross-env/-/cross-env-5.0.5.tgz#4383d364d9660873dd185b398af3bfef5efffef3" + dependencies: + cross-spawn "^5.1.0" + is-windows "^1.0.0" + +cross-spawn@^5.0.1, cross-spawn@^5.1.0: + version "5.1.0" + resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-5.1.0.tgz#e8bd0efee58fcff6f8f94510a0a554bbfa235449" + dependencies: + lru-cache "^4.0.1" + shebang-command "^1.2.0" + which "^1.2.9" + +cryptiles@2.x.x: + version "2.0.5" + resolved "https://registry.yarnpkg.com/cryptiles/-/cryptiles-2.0.5.tgz#3bdfecdc608147c1c67202fa291e7dca59eaa3b8" + dependencies: + boom "2.x.x" + +crypto-browserify@^3.11.0: + version "3.11.1" + resolved "https://registry.yarnpkg.com/crypto-browserify/-/crypto-browserify-3.11.1.tgz#948945efc6757a400d6e5e5af47194d10064279f" + dependencies: + browserify-cipher "^1.0.0" + browserify-sign "^4.0.0" + create-ecdh "^4.0.0" + create-hash "^1.1.0" + create-hmac "^1.1.0" + diffie-hellman "^5.0.0" + inherits "^2.0.1" + pbkdf2 "^3.0.3" + public-encrypt "^4.0.0" + randombytes "^2.0.0" + +css-color-names@0.0.4: + version "0.0.4" + resolved "https://registry.yarnpkg.com/css-color-names/-/css-color-names-0.0.4.tgz#808adc2e79cf84738069b646cb20ec27beb629e0" + +css-loader@^0.28.4: + version "0.28.7" + resolved "https://registry.yarnpkg.com/css-loader/-/css-loader-0.28.7.tgz#5f2ee989dd32edd907717f953317656160999c1b" + dependencies: + babel-code-frame "^6.11.0" + css-selector-tokenizer "^0.7.0" + cssnano ">=2.6.1 <4" + icss-utils "^2.1.0" + loader-utils "^1.0.2" + lodash.camelcase "^4.3.0" + object-assign "^4.0.1" + postcss "^5.0.6" + postcss-modules-extract-imports "^1.0.0" + postcss-modules-local-by-default "^1.0.1" + postcss-modules-scope "^1.0.0" + postcss-modules-values "^1.1.0" + postcss-value-parser "^3.3.0" + source-list-map "^2.0.0" + +css-select@^1.1.0, css-select@~1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/css-select/-/css-select-1.2.0.tgz#2b3a110539c5355f1cd8d314623e870b121ec858" + dependencies: + boolbase "~1.0.0" + css-what "2.1" + domutils "1.5.1" + nth-check "~1.0.1" + +css-selector-tokenizer@^0.7.0: + version "0.7.0" + resolved "https://registry.yarnpkg.com/css-selector-tokenizer/-/css-selector-tokenizer-0.7.0.tgz#e6988474ae8c953477bf5e7efecfceccd9cf4c86" + dependencies: + cssesc "^0.1.0" + fastparse "^1.1.1" + regexpu-core "^1.0.0" + +css-what@2.1: + version "2.1.0" + resolved "https://registry.yarnpkg.com/css-what/-/css-what-2.1.0.tgz#9467d032c38cfaefb9f2d79501253062f87fa1bd" + +cssesc@^0.1.0: + version "0.1.0" + resolved "https://registry.yarnpkg.com/cssesc/-/cssesc-0.1.0.tgz#c814903e45623371a0477b40109aaafbeeaddbb4" + +"cssnano@>=2.6.1 <4": + version "3.10.0" + resolved "https://registry.yarnpkg.com/cssnano/-/cssnano-3.10.0.tgz#4f38f6cea2b9b17fa01490f23f1dc68ea65c1c38" + dependencies: + autoprefixer "^6.3.1" + decamelize "^1.1.2" + defined "^1.0.0" + has "^1.0.1" + object-assign "^4.0.1" + postcss "^5.0.14" + postcss-calc "^5.2.0" + postcss-colormin "^2.1.8" + postcss-convert-values "^2.3.4" + postcss-discard-comments "^2.0.4" + postcss-discard-duplicates "^2.0.1" + postcss-discard-empty "^2.0.1" + postcss-discard-overridden "^0.1.1" + postcss-discard-unused "^2.2.1" + postcss-filter-plugins "^2.0.0" + postcss-merge-idents "^2.1.5" + postcss-merge-longhand "^2.0.1" + postcss-merge-rules "^2.0.3" + postcss-minify-font-values "^1.0.2" + postcss-minify-gradients "^1.0.1" + postcss-minify-params "^1.0.4" + postcss-minify-selectors "^2.0.4" + postcss-normalize-charset "^1.1.0" + postcss-normalize-url "^3.0.7" + postcss-ordered-values "^2.1.0" + postcss-reduce-idents "^2.2.2" + postcss-reduce-initial "^1.0.0" + postcss-reduce-transforms "^1.0.3" + postcss-svgo "^2.1.1" + postcss-unique-selectors "^2.0.2" + postcss-value-parser "^3.2.3" + postcss-zindex "^2.0.1" + +csso@~2.3.1: + version "2.3.2" + resolved "https://registry.yarnpkg.com/csso/-/csso-2.3.2.tgz#ddd52c587033f49e94b71fc55569f252e8ff5f85" + dependencies: + clap "^1.0.9" + source-map "^0.5.3" + +cssom@0.3.x, "cssom@>= 0.3.2 < 0.4.0": + version "0.3.2" + resolved "https://registry.yarnpkg.com/cssom/-/cssom-0.3.2.tgz#b8036170c79f07a90ff2f16e22284027a243848b" + +"cssstyle@>= 0.2.37 < 0.3.0": + version "0.2.37" + resolved "https://registry.yarnpkg.com/cssstyle/-/cssstyle-0.2.37.tgz#541097234cb2513c83ceed3acddc27ff27987d54" + dependencies: + cssom "0.3.x" + +currently-unhandled@^0.4.1: + version "0.4.1" + resolved "https://registry.yarnpkg.com/currently-unhandled/-/currently-unhandled-0.4.1.tgz#988df33feab191ef799a61369dd76c17adf957ea" + dependencies: + array-find-index "^1.0.1" + +d@1: + version "1.0.0" + resolved "https://registry.yarnpkg.com/d/-/d-1.0.0.tgz#754bb5bfe55451da69a58b94d45f4c5b0462d58f" + dependencies: + es5-ext "^0.10.9" + +dashdash@^1.12.0: + version "1.14.1" + resolved "https://registry.yarnpkg.com/dashdash/-/dashdash-1.14.1.tgz#853cfa0f7cbe2fed5de20326b8dd581035f6e2f0" + dependencies: + assert-plus "^1.0.0" + +date-now@^0.1.4: + version "0.1.4" + resolved "https://registry.yarnpkg.com/date-now/-/date-now-0.1.4.tgz#eaf439fd4d4848ad74e5cc7dbef200672b9e345b" + +debug@2.6.8, debug@^2.2.0, debug@^2.6.3, debug@^2.6.6, debug@^2.6.8: + version "2.6.8" + resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.8.tgz#e731531ca2ede27d188222427da17821d68ff4fc" + dependencies: + ms "2.0.0" + +decamelize@^1.0.0, decamelize@^1.1.1, decamelize@^1.1.2: + version "1.2.0" + resolved "https://registry.yarnpkg.com/decamelize/-/decamelize-1.2.0.tgz#f6534d15148269b20352e7bee26f501f9a191290" + +decode-uri-component@^0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/decode-uri-component/-/decode-uri-component-0.2.0.tgz#eb3913333458775cb84cd1a1fae062106bb87545" + +deep-equal@^1.0.1, deep-equal@~1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/deep-equal/-/deep-equal-1.0.1.tgz#f5d260292b660e084eff4cdbc9f08ad3247448b5" + +deep-extend@~0.4.0: + version "0.4.2" + resolved "https://registry.yarnpkg.com/deep-extend/-/deep-extend-0.4.2.tgz#48b699c27e334bf89f10892be432f6e4c7d34a7f" + +deep-is@~0.1.3: + version "0.1.3" + resolved "https://registry.yarnpkg.com/deep-is/-/deep-is-0.1.3.tgz#b369d6fb5dbc13eecf524f91b070feedc357cf34" + +default-require-extensions@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/default-require-extensions/-/default-require-extensions-1.0.0.tgz#f37ea15d3e13ffd9b437d33e1a75b5fb97874cb8" + dependencies: + strip-bom "^2.0.0" + +define-properties@^1.1.2: + version "1.1.2" + resolved "https://registry.yarnpkg.com/define-properties/-/define-properties-1.1.2.tgz#83a73f2fea569898fb737193c8f873caf6d45c94" + dependencies: + foreach "^2.0.5" + object-keys "^1.0.8" + +defined@^1.0.0, defined@~1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/defined/-/defined-1.0.0.tgz#c98d9bcef75674188e110969151199e39b1fa693" + +del@^2.0.2: + version "2.2.2" + resolved "https://registry.yarnpkg.com/del/-/del-2.2.2.tgz#c12c981d067846c84bcaf862cff930d907ffd1a8" + dependencies: + globby "^5.0.0" + is-path-cwd "^1.0.0" + is-path-in-cwd "^1.0.0" + object-assign "^4.0.1" + pify "^2.0.0" + pinkie-promise "^2.0.0" + rimraf "^2.2.8" + +del@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/del/-/del-3.0.0.tgz#53ecf699ffcbcb39637691ab13baf160819766e5" + dependencies: + globby "^6.1.0" + is-path-cwd "^1.0.0" + is-path-in-cwd "^1.0.0" + p-map "^1.1.1" + pify "^3.0.0" + rimraf "^2.2.8" + +delayed-stream@~1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/delayed-stream/-/delayed-stream-1.0.0.tgz#df3ae199acadfb7d440aaae0b29e2272b24ec619" + +delegates@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/delegates/-/delegates-1.0.0.tgz#84c6e159b81904fdca59a0ef44cd870d31250f9a" + +depd@1.1.1, depd@~1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/depd/-/depd-1.1.1.tgz#5783b4e1c459f06fa5ca27f991f3d06e7a310359" + +des.js@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/des.js/-/des.js-1.0.0.tgz#c074d2e2aa6a8a9a07dbd61f9a15c2cd83ec8ecc" + dependencies: + inherits "^2.0.1" + minimalistic-assert "^1.0.0" + +destroy@~1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/destroy/-/destroy-1.0.4.tgz#978857442c44749e4206613e37946205826abd80" + +detect-indent@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/detect-indent/-/detect-indent-4.0.0.tgz#f76d064352cdf43a1cb6ce619c4ee3a9475de208" + dependencies: + repeating "^2.0.0" + +detect-node@^2.0.3: + version "2.0.3" + resolved "https://registry.yarnpkg.com/detect-node/-/detect-node-2.0.3.tgz#a2033c09cc8e158d37748fbde7507832bd6ce127" + +diff@^3.1.0, diff@^3.2.0: + version "3.3.1" + resolved "https://registry.yarnpkg.com/diff/-/diff-3.3.1.tgz#aa8567a6eed03c531fc89d3f711cd0e5259dec75" + +diffie-hellman@^5.0.0: + version "5.0.2" + resolved "https://registry.yarnpkg.com/diffie-hellman/-/diffie-hellman-5.0.2.tgz#b5835739270cfe26acf632099fded2a07f209e5e" + dependencies: + bn.js "^4.1.0" + miller-rabin "^4.0.0" + randombytes "^2.0.0" + +dns-equal@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/dns-equal/-/dns-equal-1.0.0.tgz#b39e7f1da6eb0a75ba9c17324b34753c47e0654d" + +dns-packet@^1.0.1: + version "1.2.2" + resolved "https://registry.yarnpkg.com/dns-packet/-/dns-packet-1.2.2.tgz#a8a26bec7646438963fc86e06f8f8b16d6c8bf7a" + dependencies: + ip "^1.1.0" + safe-buffer "^5.0.1" + +dns-txt@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/dns-txt/-/dns-txt-2.0.2.tgz#b91d806f5d27188e4ab3e7d107d881a1cc4642b6" + dependencies: + buffer-indexof "^1.0.0" + +doctrine@1.5.0: + version "1.5.0" + resolved "https://registry.yarnpkg.com/doctrine/-/doctrine-1.5.0.tgz#379dce730f6166f76cefa4e6707a159b02c5a6fa" + dependencies: + esutils "^2.0.2" + isarray "^1.0.0" + +doctrine@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/doctrine/-/doctrine-2.0.0.tgz#c73d8d2909d22291e1a007a395804da8b665fe63" + dependencies: + esutils "^2.0.2" + isarray "^1.0.0" + +dom-converter@~0.1: + version "0.1.4" + resolved "https://registry.yarnpkg.com/dom-converter/-/dom-converter-0.1.4.tgz#a45ef5727b890c9bffe6d7c876e7b19cb0e17f3b" + dependencies: + utila "~0.3" + +dom-helpers@^3.2.0: + version "3.2.1" + resolved "https://registry.yarnpkg.com/dom-helpers/-/dom-helpers-3.2.1.tgz#3203e07fed217bd1f424b019735582fc37b2825a" + +dom-serializer@0, dom-serializer@~0.1.0: + version "0.1.0" + resolved "https://registry.yarnpkg.com/dom-serializer/-/dom-serializer-0.1.0.tgz#073c697546ce0780ce23be4a28e293e40bc30c82" + dependencies: + domelementtype "~1.1.1" + entities "~1.1.1" + +domain-browser@^1.1.1: + version "1.1.7" + resolved "https://registry.yarnpkg.com/domain-browser/-/domain-browser-1.1.7.tgz#867aa4b093faa05f1de08c06f4d7b21fdf8698bc" + +domelementtype@1, domelementtype@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/domelementtype/-/domelementtype-1.3.0.tgz#b17aed82e8ab59e52dd9c19b1756e0fc187204c2" + +domelementtype@~1.1.1: + version "1.1.3" + resolved "https://registry.yarnpkg.com/domelementtype/-/domelementtype-1.1.3.tgz#bd28773e2642881aec51544924299c5cd822185b" + +domhandler@2.1: + version "2.1.0" + resolved "https://registry.yarnpkg.com/domhandler/-/domhandler-2.1.0.tgz#d2646f5e57f6c3bab11cf6cb05d3c0acf7412594" + dependencies: + domelementtype "1" + +domhandler@^2.3.0: + version "2.4.1" + resolved "https://registry.yarnpkg.com/domhandler/-/domhandler-2.4.1.tgz#892e47000a99be55bbf3774ffea0561d8879c259" + dependencies: + domelementtype "1" + +domutils@1.1: + version "1.1.6" + resolved "https://registry.yarnpkg.com/domutils/-/domutils-1.1.6.tgz#bddc3de099b9a2efacc51c623f28f416ecc57485" + dependencies: + domelementtype "1" + +domutils@1.5.1: + version "1.5.1" + resolved "https://registry.yarnpkg.com/domutils/-/domutils-1.5.1.tgz#dcd8488a26f563d61079e48c9f7b7e32373682cf" + dependencies: + dom-serializer "0" + domelementtype "1" + +domutils@^1.5.1: + version "1.6.2" + resolved "https://registry.yarnpkg.com/domutils/-/domutils-1.6.2.tgz#1958cc0b4c9426e9ed367fb1c8e854891b0fa3ff" + dependencies: + dom-serializer "0" + domelementtype "1" + +dotenv@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/dotenv/-/dotenv-4.0.0.tgz#864ef1379aced55ce6f95debecdce179f7a0cd1d" + +"drone-js@file:./vendor/drone-js": + version "0.0.0" + +ecc-jsbn@~0.1.1: + version "0.1.1" + resolved "https://registry.yarnpkg.com/ecc-jsbn/-/ecc-jsbn-0.1.1.tgz#0fc73a9ed5f0d53c38193398523ef7e543777505" + dependencies: + jsbn "~0.1.0" + +ee-first@1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/ee-first/-/ee-first-1.1.1.tgz#590c61156b0ae2f4f0255732a158b266bc56b21d" + +electron-to-chromium@^1.2.7, electron-to-chromium@^1.3.18: + version "1.3.21" + resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.3.21.tgz#a967ebdcfe8ed0083fc244d1894022a8e8113ea2" + +elliptic@^6.0.0: + version "6.4.0" + resolved "https://registry.yarnpkg.com/elliptic/-/elliptic-6.4.0.tgz#cac9af8762c85836187003c8dfe193e5e2eae5df" + dependencies: + bn.js "^4.4.0" + brorand "^1.0.1" + hash.js "^1.0.0" + hmac-drbg "^1.0.0" + inherits "^2.0.1" + minimalistic-assert "^1.0.0" + minimalistic-crypto-utils "^1.0.0" + +emmett@^3.1.1: + version "3.1.1" + resolved "https://registry.yarnpkg.com/emmett/-/emmett-3.1.1.tgz#7173858aabdfece0f40232696d4b13fef232cf04" + +emojis-list@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/emojis-list/-/emojis-list-2.1.0.tgz#4daa4d9db00f9819880c79fa457ae5b09a1fd389" + +encodeurl@~1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/encodeurl/-/encodeurl-1.0.1.tgz#79e3d58655346909fe6f0f45a5de68103b294d20" + +encoding@^0.1.11: + version "0.1.12" + resolved "https://registry.yarnpkg.com/encoding/-/encoding-0.1.12.tgz#538b66f3ee62cd1ab51ec323829d1f9480c74beb" + dependencies: + iconv-lite "~0.4.13" + +enhanced-resolve@^3.4.0: + version "3.4.1" + resolved "https://registry.yarnpkg.com/enhanced-resolve/-/enhanced-resolve-3.4.1.tgz#0421e339fd71419b3da13d129b3979040230476e" + dependencies: + graceful-fs "^4.1.2" + memory-fs "^0.4.0" + object-assign "^4.0.1" + tapable "^0.2.7" + +entities@^1.1.1, entities@~1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/entities/-/entities-1.1.1.tgz#6e5c2d0a5621b5dadaecef80b90edfb5cd7772f0" + +enzyme@^2.9.1: + version "2.9.1" + resolved "https://registry.yarnpkg.com/enzyme/-/enzyme-2.9.1.tgz#07d5ce691241240fb817bf2c4b18d6e530240df6" + dependencies: + cheerio "^0.22.0" + function.prototype.name "^1.0.0" + is-subset "^0.1.1" + lodash "^4.17.4" + object-is "^1.0.1" + object.assign "^4.0.4" + object.entries "^1.0.4" + object.values "^1.0.4" + prop-types "^15.5.10" + uuid "^3.0.1" + +errno@^0.1.1, errno@^0.1.3, errno@^0.1.4: + version "0.1.4" + resolved "https://registry.yarnpkg.com/errno/-/errno-0.1.4.tgz#b896e23a9e5e8ba33871fc996abd3635fc9a1c7d" + dependencies: + prr "~0.0.0" + +error-ex@^1.2.0: + version "1.3.1" + resolved "https://registry.yarnpkg.com/error-ex/-/error-ex-1.3.1.tgz#f855a86ce61adc4e8621c3cda21e7a7612c3a8dc" + dependencies: + is-arrayish "^0.2.1" + +es-abstract@^1.5.0, es-abstract@^1.6.1, es-abstract@^1.7.0: + version "1.8.2" + resolved "https://registry.yarnpkg.com/es-abstract/-/es-abstract-1.8.2.tgz#25103263dc4decbda60e0c737ca32313518027ee" + dependencies: + es-to-primitive "^1.1.1" + function-bind "^1.1.1" + has "^1.0.1" + is-callable "^1.1.3" + is-regex "^1.0.4" + +es-to-primitive@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/es-to-primitive/-/es-to-primitive-1.1.1.tgz#45355248a88979034b6792e19bb81f2b7975dd0d" + dependencies: + is-callable "^1.1.1" + is-date-object "^1.0.1" + is-symbol "^1.0.1" + +es5-ext@^0.10.14, es5-ext@^0.10.9, es5-ext@~0.10.14: + version "0.10.30" + resolved "https://registry.yarnpkg.com/es5-ext/-/es5-ext-0.10.30.tgz#7141a16836697dbabfaaaeee41495ce29f52c939" + dependencies: + es6-iterator "2" + es6-symbol "~3.1" + +es6-iterator@2, es6-iterator@^2.0.1, es6-iterator@~2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/es6-iterator/-/es6-iterator-2.0.1.tgz#8e319c9f0453bf575d374940a655920e59ca5512" + dependencies: + d "1" + es5-ext "^0.10.14" + es6-symbol "^3.1" + +es6-map@^0.1.3: + version "0.1.5" + resolved "https://registry.yarnpkg.com/es6-map/-/es6-map-0.1.5.tgz#9136e0503dcc06a301690f0bb14ff4e364e949f0" + dependencies: + d "1" + es5-ext "~0.10.14" + es6-iterator "~2.0.1" + es6-set "~0.1.5" + es6-symbol "~3.1.1" + event-emitter "~0.3.5" + +es6-set@~0.1.5: + version "0.1.5" + resolved "https://registry.yarnpkg.com/es6-set/-/es6-set-0.1.5.tgz#d2b3ec5d4d800ced818db538d28974db0a73ccb1" + dependencies: + d "1" + es5-ext "~0.10.14" + es6-iterator "~2.0.1" + es6-symbol "3.1.1" + event-emitter "~0.3.5" + +es6-symbol@3.1.1, es6-symbol@^3.1, es6-symbol@^3.1.1, es6-symbol@~3.1, es6-symbol@~3.1.1: + version "3.1.1" + resolved "https://registry.yarnpkg.com/es6-symbol/-/es6-symbol-3.1.1.tgz#bf00ef4fdab6ba1b46ecb7b629b4c7ed5715cc77" + dependencies: + d "1" + es5-ext "~0.10.14" + +es6-weak-map@^2.0.1: + version "2.0.2" + resolved "https://registry.yarnpkg.com/es6-weak-map/-/es6-weak-map-2.0.2.tgz#5e3ab32251ffd1538a1f8e5ffa1357772f92d96f" + dependencies: + d "1" + es5-ext "^0.10.14" + es6-iterator "^2.0.1" + es6-symbol "^3.1.1" + +escape-html@~1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/escape-html/-/escape-html-1.0.3.tgz#0258eae4d3d0c0974de1c169188ef0051d1d1988" + +escape-string-regexp@^1.0.2, escape-string-regexp@^1.0.5, escape-string-regexp@~1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz#1b61c0562190a8dff6ae3bb2cf0200ca130b86d4" + +escodegen@^1.6.1: + version "1.9.0" + resolved "https://registry.yarnpkg.com/escodegen/-/escodegen-1.9.0.tgz#9811a2f265dc1cd3894420ee3717064b632b8852" + dependencies: + esprima "^3.1.3" + estraverse "^4.2.0" + esutils "^2.0.2" + optionator "^0.8.1" + optionalDependencies: + source-map "~0.5.6" + +escope@^3.6.0: + version "3.6.0" + resolved "https://registry.yarnpkg.com/escope/-/escope-3.6.0.tgz#e01975e812781a163a6dadfdd80398dc64c889c3" + dependencies: + es6-map "^0.1.3" + es6-weak-map "^2.0.1" + esrecurse "^4.1.0" + estraverse "^4.1.1" + +eslint: + version "4.6.1" + resolved "https://registry.yarnpkg.com/eslint/-/eslint-4.6.1.tgz#ddc7fc7fd70bf93205b0b3449bb16a1e9e7d4950" + dependencies: + ajv "^5.2.0" + babel-code-frame "^6.22.0" + chalk "^2.1.0" + concat-stream "^1.6.0" + cross-spawn "^5.1.0" + debug "^2.6.8" + doctrine "^2.0.0" + eslint-scope "^3.7.1" + espree "^3.5.0" + esquery "^1.0.0" + estraverse "^4.2.0" + esutils "^2.0.2" + file-entry-cache "^2.0.0" + functional-red-black-tree "^1.0.1" + glob "^7.1.2" + globals "^9.17.0" + ignore "^3.3.3" + imurmurhash "^0.1.4" + inquirer "^3.0.6" + is-resolvable "^1.0.0" + js-yaml "^3.9.1" + json-stable-stringify "^1.0.1" + levn "^0.3.0" + lodash "^4.17.4" + minimatch "^3.0.2" + mkdirp "^0.5.1" + natural-compare "^1.4.0" + optionator "^0.8.2" + path-is-inside "^1.0.2" + pluralize "^4.0.0" + progress "^2.0.0" + require-uncached "^1.0.3" + semver "^5.3.0" + strip-ansi "^4.0.0" + strip-json-comments "~2.0.1" + table "^4.0.1" + text-table "~0.2.0" + +eslint-config-prettier@^2.4.0: + version "2.4.0" + resolved "https://registry.yarnpkg.com/eslint-config-prettier/-/eslint-config-prettier-2.4.0.tgz#fb7cf29c0ab2ba61af5164fb1930f9bef3be2872" + dependencies: + get-stdin "^5.0.1" + +eslint-config-standard@^10.2.1: + version "10.2.1" + resolved "https://registry.yarnpkg.com/eslint-config-standard/-/eslint-config-standard-10.2.1.tgz#c061e4d066f379dc17cd562c64e819b4dd454591" + +eslint-import-resolver-node@^0.3.1: + version "0.3.1" + resolved "https://registry.yarnpkg.com/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.1.tgz#4422574cde66a9a7b099938ee4d508a199e0e3cc" + dependencies: + debug "^2.6.8" + resolve "^1.2.0" + +eslint-module-utils@^2.1.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/eslint-module-utils/-/eslint-module-utils-2.1.1.tgz#abaec824177613b8a95b299639e1b6facf473449" + dependencies: + debug "^2.6.8" + pkg-dir "^1.0.0" + +eslint-plugin-import@^2.7.0: + version "2.7.0" + resolved "https://registry.yarnpkg.com/eslint-plugin-import/-/eslint-plugin-import-2.7.0.tgz#21de33380b9efb55f5ef6d2e210ec0e07e7fa69f" + dependencies: + builtin-modules "^1.1.1" + contains-path "^0.1.0" + debug "^2.6.8" + doctrine "1.5.0" + eslint-import-resolver-node "^0.3.1" + eslint-module-utils "^2.1.1" + has "^1.0.1" + lodash.cond "^4.3.0" + minimatch "^3.0.3" + read-pkg-up "^2.0.0" + +eslint-plugin-jest: + version "21.0.2" + resolved "https://registry.yarnpkg.com/eslint-plugin-jest/-/eslint-plugin-jest-21.0.2.tgz#2eba7745844648513889d9d1bbefa3f5e3d05e54" + +eslint-plugin-node@^5.1.1: + version "5.1.1" + resolved "https://registry.yarnpkg.com/eslint-plugin-node/-/eslint-plugin-node-5.1.1.tgz#a7ed956e780c22aef6afd1116005acd82f26eac6" + dependencies: + ignore "^3.3.3" + minimatch "^3.0.4" + resolve "^1.3.3" + semver "5.3.0" + +eslint-plugin-prettier@^2.2.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/eslint-plugin-prettier/-/eslint-plugin-prettier-2.2.0.tgz#f2837ad063903d73c621e7188fb3d41486434088" + dependencies: + fast-diff "^1.1.1" + jest-docblock "^20.0.1" + +eslint-plugin-promise@^3.5.0: + version "3.5.0" + resolved "https://registry.yarnpkg.com/eslint-plugin-promise/-/eslint-plugin-promise-3.5.0.tgz#78fbb6ffe047201627569e85a6c5373af2a68fca" + +eslint-plugin-react: + version "7.3.0" + resolved "https://registry.yarnpkg.com/eslint-plugin-react/-/eslint-plugin-react-7.3.0.tgz#ca9368da36f733fbdc05718ae4e91f778f38e344" + dependencies: + doctrine "^2.0.0" + has "^1.0.1" + jsx-ast-utils "^2.0.0" + prop-types "^15.5.10" + +eslint-plugin-standard@^3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/eslint-plugin-standard/-/eslint-plugin-standard-3.0.1.tgz#34d0c915b45edc6f010393c7eef3823b08565cf2" + +eslint-scope@^3.7.1: + version "3.7.1" + resolved "https://registry.yarnpkg.com/eslint-scope/-/eslint-scope-3.7.1.tgz#3d63c3edfda02e06e01a452ad88caacc7cdcb6e8" + dependencies: + esrecurse "^4.1.0" + estraverse "^4.1.1" + +espree@^3.5.0: + version "3.5.0" + resolved "https://registry.yarnpkg.com/espree/-/espree-3.5.0.tgz#98358625bdd055861ea27e2867ea729faf463d8d" + dependencies: + acorn "^5.1.1" + acorn-jsx "^3.0.0" + +esprima@^2.6.0: + version "2.7.3" + resolved "https://registry.yarnpkg.com/esprima/-/esprima-2.7.3.tgz#96e3b70d5779f6ad49cd032673d1c312767ba581" + +esprima@^3.1.3: + version "3.1.3" + resolved "https://registry.yarnpkg.com/esprima/-/esprima-3.1.3.tgz#fdca51cee6133895e3c88d535ce49dbff62a4633" + +esprima@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/esprima/-/esprima-4.0.0.tgz#4499eddcd1110e0b218bacf2fa7f7f59f55ca804" + +esquery@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/esquery/-/esquery-1.0.0.tgz#cfba8b57d7fba93f17298a8a006a04cda13d80fa" + dependencies: + estraverse "^4.0.0" + +esrecurse@^4.1.0: + version "4.2.0" + resolved "https://registry.yarnpkg.com/esrecurse/-/esrecurse-4.2.0.tgz#fa9568d98d3823f9a41d91e902dcab9ea6e5b163" + dependencies: + estraverse "^4.1.0" + object-assign "^4.0.1" + +estraverse@^4.0.0, estraverse@^4.1.0, estraverse@^4.1.1, estraverse@^4.2.0: + version "4.2.0" + resolved "https://registry.yarnpkg.com/estraverse/-/estraverse-4.2.0.tgz#0dee3fed31fcd469618ce7342099fc1afa0bdb13" + +esutils@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/esutils/-/esutils-2.0.2.tgz#0abf4f1caa5bcb1f7a9d8acc6dea4faaa04bac9b" + +etag@~1.8.0: + version "1.8.0" + resolved "https://registry.yarnpkg.com/etag/-/etag-1.8.0.tgz#6f631aef336d6c46362b51764044ce216be3c051" + +event-emitter@~0.3.5: + version "0.3.5" + resolved "https://registry.yarnpkg.com/event-emitter/-/event-emitter-0.3.5.tgz#df8c69eef1647923c7157b9ce83840610b02cc39" + dependencies: + d "1" + es5-ext "~0.10.14" + +eventemitter3@1.x.x: + version "1.2.0" + resolved "https://registry.yarnpkg.com/eventemitter3/-/eventemitter3-1.2.0.tgz#1c86991d816ad1e504750e73874224ecf3bec508" + +events@^1.0.0: + version "1.1.1" + resolved "https://registry.yarnpkg.com/events/-/events-1.1.1.tgz#9ebdb7635ad099c70dcc4c2a1f5004288e8bd924" + +eventsource@0.1.6: + version "0.1.6" + resolved "https://registry.yarnpkg.com/eventsource/-/eventsource-0.1.6.tgz#0acede849ed7dd1ccc32c811bb11b944d4f29232" + dependencies: + original ">=0.0.5" + +evp_bytestokey@^1.0.0, evp_bytestokey@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/evp_bytestokey/-/evp_bytestokey-1.0.3.tgz#7fcbdb198dc71959432efe13842684e0525acb02" + dependencies: + md5.js "^1.3.4" + safe-buffer "^5.1.1" + +exec-sh@^0.2.0: + version "0.2.1" + resolved "https://registry.yarnpkg.com/exec-sh/-/exec-sh-0.2.1.tgz#163b98a6e89e6b65b47c2a28d215bc1f63989c38" + dependencies: + merge "^1.1.3" + +execa@^0.7.0: + version "0.7.0" + resolved "https://registry.yarnpkg.com/execa/-/execa-0.7.0.tgz#944becd34cc41ee32a63a9faf27ad5a65fc59777" + dependencies: + cross-spawn "^5.0.1" + get-stream "^3.0.0" + is-stream "^1.1.0" + npm-run-path "^2.0.0" + p-finally "^1.0.0" + signal-exit "^3.0.0" + strip-eof "^1.0.0" + +expand-brackets@^0.1.4: + version "0.1.5" + resolved "https://registry.yarnpkg.com/expand-brackets/-/expand-brackets-0.1.5.tgz#df07284e342a807cd733ac5af72411e581d1177b" + dependencies: + is-posix-bracket "^0.1.0" + +expand-range@^1.8.1: + version "1.8.2" + resolved "https://registry.yarnpkg.com/expand-range/-/expand-range-1.8.2.tgz#a299effd335fe2721ebae8e257ec79644fc85337" + dependencies: + fill-range "^2.1.0" + +expect@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/expect/-/expect-21.0.2.tgz#b34abf0635ec9d6aea1ce7edb4722afe86c4a38f" + dependencies: + ansi-styles "^3.2.0" + jest-diff "^21.0.2" + jest-get-type "^21.0.2" + jest-matcher-utils "^21.0.2" + jest-message-util "^21.0.2" + jest-regex-util "^21.0.2" + +express@^4.13.3: + version "4.15.4" + resolved "https://registry.yarnpkg.com/express/-/express-4.15.4.tgz#032e2253489cf8fce02666beca3d11ed7a2daed1" + dependencies: + accepts "~1.3.3" + array-flatten "1.1.1" + content-disposition "0.5.2" + content-type "~1.0.2" + cookie "0.3.1" + cookie-signature "1.0.6" + debug "2.6.8" + depd "~1.1.1" + encodeurl "~1.0.1" + escape-html "~1.0.3" + etag "~1.8.0" + finalhandler "~1.0.4" + fresh "0.5.0" + merge-descriptors "1.0.1" + methods "~1.1.2" + on-finished "~2.3.0" + parseurl "~1.3.1" + path-to-regexp "0.1.7" + proxy-addr "~1.1.5" + qs "6.5.0" + range-parser "~1.2.0" + send "0.15.4" + serve-static "1.12.4" + setprototypeof "1.0.3" + statuses "~1.3.1" + type-is "~1.6.15" + utils-merge "1.0.0" + vary "~1.1.1" + +extend@~3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/extend/-/extend-3.0.1.tgz#a755ea7bc1adfcc5a31ce7e762dbaadc5e636444" + +external-editor@^2.0.4: + version "2.0.4" + resolved "https://registry.yarnpkg.com/external-editor/-/external-editor-2.0.4.tgz#1ed9199da9cbfe2ef2f7a31b2fde8b0d12368972" + dependencies: + iconv-lite "^0.4.17" + jschardet "^1.4.2" + tmp "^0.0.31" + +extglob@^0.3.1: + version "0.3.2" + resolved "https://registry.yarnpkg.com/extglob/-/extglob-0.3.2.tgz#2e18ff3d2f49ab2765cec9023f011daa8d8349a1" + dependencies: + is-extglob "^1.0.0" + +extsprintf@1.3.0, extsprintf@^1.2.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/extsprintf/-/extsprintf-1.3.0.tgz#96918440e3041a7a414f8c52e3c574eb3c3e1e05" + +fast-deep-equal@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-1.0.0.tgz#96256a3bc975595eb36d82e9929d060d893439ff" + +fast-diff@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/fast-diff/-/fast-diff-1.1.1.tgz#0aea0e4e605b6a2189f0e936d4b7fbaf1b7cfd9b" + +fast-levenshtein@~2.0.4: + version "2.0.6" + resolved "https://registry.yarnpkg.com/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz#3d8a5c66883a16a30ca8643e851f19baa7797917" + +fastparse@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/fastparse/-/fastparse-1.1.1.tgz#d1e2643b38a94d7583b479060e6c4affc94071f8" + +faye-websocket@^0.10.0: + version "0.10.0" + resolved "https://registry.yarnpkg.com/faye-websocket/-/faye-websocket-0.10.0.tgz#4e492f8d04dfb6f89003507f6edbf2d501e7c6f4" + dependencies: + websocket-driver ">=0.5.1" + +faye-websocket@~0.11.0: + version "0.11.1" + resolved "https://registry.yarnpkg.com/faye-websocket/-/faye-websocket-0.11.1.tgz#f0efe18c4f56e4f40afc7e06c719fd5ee6188f38" + dependencies: + websocket-driver ">=0.5.1" + +fb-watchman@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/fb-watchman/-/fb-watchman-2.0.0.tgz#54e9abf7dfa2f26cd9b1636c588c1afc05de5d58" + dependencies: + bser "^2.0.0" + +fbjs@^0.8.9: + version "0.8.15" + resolved "https://registry.yarnpkg.com/fbjs/-/fbjs-0.8.15.tgz#4f0695fdfcc16c37c0b07facec8cb4c4091685b9" + dependencies: + core-js "^1.0.0" + isomorphic-fetch "^2.1.1" + loose-envify "^1.0.0" + object-assign "^4.1.0" + promise "^7.1.1" + setimmediate "^1.0.5" + ua-parser-js "^0.7.9" + +figures@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/figures/-/figures-2.0.0.tgz#3ab1a2d2a62c8bfb431a0c94cb797a2fce27c962" + dependencies: + escape-string-regexp "^1.0.5" + +file-entry-cache@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/file-entry-cache/-/file-entry-cache-2.0.0.tgz#c392990c3e684783d838b8c84a45d8a048458361" + dependencies: + flat-cache "^1.2.1" + object-assign "^4.0.1" + +file-loader@^0.11.2: + version "0.11.2" + resolved "https://registry.yarnpkg.com/file-loader/-/file-loader-0.11.2.tgz#4ff1df28af38719a6098093b88c82c71d1794a34" + dependencies: + loader-utils "^1.0.2" + +filename-regex@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/filename-regex/-/filename-regex-2.0.1.tgz#c1c4b9bee3e09725ddb106b75c1e301fe2f18b26" + +fileset@^2.0.2: + version "2.0.3" + resolved "https://registry.yarnpkg.com/fileset/-/fileset-2.0.3.tgz#8e7548a96d3cc2327ee5e674168723a333bba2a0" + dependencies: + glob "^7.0.3" + minimatch "^3.0.3" + +fill-range@^2.1.0: + version "2.2.3" + resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-2.2.3.tgz#50b77dfd7e469bc7492470963699fe7a8485a723" + dependencies: + is-number "^2.1.0" + isobject "^2.0.0" + randomatic "^1.1.3" + repeat-element "^1.1.2" + repeat-string "^1.5.2" + +finalhandler@~1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/finalhandler/-/finalhandler-1.0.4.tgz#18574f2e7c4b98b8ae3b230c21f201f31bdb3fb7" + dependencies: + debug "2.6.8" + encodeurl "~1.0.1" + escape-html "~1.0.3" + on-finished "~2.3.0" + parseurl "~1.3.1" + statuses "~1.3.1" + unpipe "~1.0.0" + +find-cache-dir@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/find-cache-dir/-/find-cache-dir-1.0.0.tgz#9288e3e9e3cc3748717d39eade17cf71fc30ee6f" + dependencies: + commondir "^1.0.1" + make-dir "^1.0.0" + pkg-dir "^2.0.0" + +find-up@^1.0.0: + version "1.1.2" + resolved "https://registry.yarnpkg.com/find-up/-/find-up-1.1.2.tgz#6b2e9822b1a2ce0a60ab64d610eccad53cb24d0f" + dependencies: + path-exists "^2.0.0" + pinkie-promise "^2.0.0" + +find-up@^2.0.0, find-up@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/find-up/-/find-up-2.1.0.tgz#45d1b7e506c717ddd482775a2b77920a3c0c57a7" + dependencies: + locate-path "^2.0.0" + +flat-cache@^1.2.1: + version "1.2.2" + resolved "https://registry.yarnpkg.com/flat-cache/-/flat-cache-1.2.2.tgz#fa86714e72c21db88601761ecf2f555d1abc6b96" + dependencies: + circular-json "^0.3.1" + del "^2.0.2" + graceful-fs "^4.1.2" + write "^0.2.1" + +flatten@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/flatten/-/flatten-1.0.2.tgz#dae46a9d78fbe25292258cc1e780a41d95c03782" + +for-each@~0.3.2: + version "0.3.2" + resolved "https://registry.yarnpkg.com/for-each/-/for-each-0.3.2.tgz#2c40450b9348e97f281322593ba96704b9abd4d4" + dependencies: + is-function "~1.0.0" + +for-in@^1.0.1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/for-in/-/for-in-1.0.2.tgz#81068d295a8142ec0ac726c6e2200c30fb6d5e80" + +for-own@^0.1.4: + version "0.1.5" + resolved "https://registry.yarnpkg.com/for-own/-/for-own-0.1.5.tgz#5265c681a4f294dabbf17c9509b6763aa84510ce" + dependencies: + for-in "^1.0.1" + +foreach@^2.0.5: + version "2.0.5" + resolved "https://registry.yarnpkg.com/foreach/-/foreach-2.0.5.tgz#0bee005018aeb260d0a3af3ae658dd0136ec1b99" + +forever-agent@~0.6.1: + version "0.6.1" + resolved "https://registry.yarnpkg.com/forever-agent/-/forever-agent-0.6.1.tgz#fbc71f0c41adeb37f96c577ad1ed42d8fdacca91" + +form-data@~2.1.1: + version "2.1.4" + resolved "https://registry.yarnpkg.com/form-data/-/form-data-2.1.4.tgz#33c183acf193276ecaa98143a69e94bfee1750d1" + dependencies: + asynckit "^0.4.0" + combined-stream "^1.0.5" + mime-types "^2.1.12" + +formatio@1.2.0, formatio@^1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/formatio/-/formatio-1.2.0.tgz#f3b2167d9068c4698a8d51f4f760a39a54d818eb" + dependencies: + samsam "1.x" + +forwarded@~0.1.0: + version "0.1.0" + resolved "https://registry.yarnpkg.com/forwarded/-/forwarded-0.1.0.tgz#19ef9874c4ae1c297bcf078fde63a09b66a84363" + +fresh@0.5.0: + version "0.5.0" + resolved "https://registry.yarnpkg.com/fresh/-/fresh-0.5.0.tgz#f474ca5e6a9246d6fd8e0953cfa9b9c805afa78e" + +fs.realpath@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f" + +fsevents@^1.0.0, fsevents@^1.1.1: + version "1.1.2" + resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-1.1.2.tgz#3282b713fb3ad80ede0e9fcf4611b5aa6fc033f4" + dependencies: + nan "^2.3.0" + node-pre-gyp "^0.6.36" + +fstream-ignore@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/fstream-ignore/-/fstream-ignore-1.0.5.tgz#9c31dae34767018fe1d249b24dada67d092da105" + dependencies: + fstream "^1.0.0" + inherits "2" + minimatch "^3.0.0" + +fstream@^1.0.0, fstream@^1.0.10, fstream@^1.0.2: + version "1.0.11" + resolved "https://registry.yarnpkg.com/fstream/-/fstream-1.0.11.tgz#5c1fb1f117477114f0632a0eb4b71b3cb0fd3171" + dependencies: + graceful-fs "^4.1.2" + inherits "~2.0.0" + mkdirp ">=0.5 0" + rimraf "2" + +function-bind@^1.0.2, function-bind@^1.1.0, function-bind@^1.1.1, function-bind@~1.1.0: + version "1.1.1" + resolved "https://registry.yarnpkg.com/function-bind/-/function-bind-1.1.1.tgz#a56899d3ea3c9bab874bb9773b7c5ede92f4895d" + +function.prototype.name@^1.0.0: + version "1.0.3" + resolved "https://registry.yarnpkg.com/function.prototype.name/-/function.prototype.name-1.0.3.tgz#0099ae5572e9dd6f03c97d023fd92bcc5e639eac" + dependencies: + define-properties "^1.1.2" + function-bind "^1.1.0" + is-callable "^1.1.3" + +functional-red-black-tree@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz#1b0ab3bd553b2a0d6399d29c0e3ea0b252078327" + +gauge@~2.7.3: + version "2.7.4" + resolved "https://registry.yarnpkg.com/gauge/-/gauge-2.7.4.tgz#2c03405c7538c39d7eb37b317022e325fb018bf7" + dependencies: + aproba "^1.0.3" + console-control-strings "^1.0.0" + has-unicode "^2.0.0" + object-assign "^4.1.0" + signal-exit "^3.0.0" + string-width "^1.0.1" + strip-ansi "^3.0.1" + wide-align "^1.1.0" + +get-caller-file@^1.0.1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/get-caller-file/-/get-caller-file-1.0.2.tgz#f702e63127e7e231c160a80c1554acb70d5047e5" + +get-stdin@^4.0.1: + version "4.0.1" + resolved "https://registry.yarnpkg.com/get-stdin/-/get-stdin-4.0.1.tgz#b968c6b0a04384324902e8bf1a5df32579a450fe" + +get-stdin@^5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/get-stdin/-/get-stdin-5.0.1.tgz#122e161591e21ff4c52530305693f20e6393a398" + +get-stream@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-3.0.0.tgz#8e943d1358dc37555054ecbe2edb05aa174ede14" + +getpass@^0.1.1: + version "0.1.7" + resolved "https://registry.yarnpkg.com/getpass/-/getpass-0.1.7.tgz#5eff8e3e684d569ae4cb2b1282604e8ba62149fa" + dependencies: + assert-plus "^1.0.0" + +glob-base@^0.3.0: + version "0.3.0" + resolved "https://registry.yarnpkg.com/glob-base/-/glob-base-0.3.0.tgz#dbb164f6221b1c0b1ccf82aea328b497df0ea3c4" + dependencies: + glob-parent "^2.0.0" + is-glob "^2.0.0" + +glob-parent@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-2.0.0.tgz#81383d72db054fcccf5336daa902f182f6edbb28" + dependencies: + is-glob "^2.0.0" + +glob@^7.0.3, glob@^7.0.5, glob@^7.1.1, glob@^7.1.2, glob@~7.1.2: + version "7.1.2" + resolved "https://registry.yarnpkg.com/glob/-/glob-7.1.2.tgz#c19c9df9a028702d678612384a6552404c636d15" + dependencies: + fs.realpath "^1.0.0" + inflight "^1.0.4" + inherits "2" + minimatch "^3.0.4" + once "^1.3.0" + path-is-absolute "^1.0.0" + +globals@^9.17.0, globals@^9.18.0: + version "9.18.0" + resolved "https://registry.yarnpkg.com/globals/-/globals-9.18.0.tgz#aa3896b3e69b487f17e31ed2143d69a8e30c2d8a" + +globby@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/globby/-/globby-5.0.0.tgz#ebd84667ca0dbb330b99bcfc68eac2bc54370e0d" + dependencies: + array-union "^1.0.1" + arrify "^1.0.0" + glob "^7.0.3" + object-assign "^4.0.1" + pify "^2.0.0" + pinkie-promise "^2.0.0" + +globby@^6.1.0: + version "6.1.0" + resolved "https://registry.yarnpkg.com/globby/-/globby-6.1.0.tgz#f5a6d70e8395e21c858fb0489d64df02424d506c" + dependencies: + array-union "^1.0.1" + glob "^7.0.3" + object-assign "^4.0.1" + pify "^2.0.0" + pinkie-promise "^2.0.0" + +graceful-fs@^4.1.11, graceful-fs@^4.1.2, graceful-fs@^4.1.9: + version "4.1.11" + resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.1.11.tgz#0e8bdfe4d1ddb8854d64e04ea7c00e2a026e5658" + +growly@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/growly/-/growly-1.3.0.tgz#f10748cbe76af964b7c96c93c6bcc28af120c081" + +handle-thing@^1.2.5: + version "1.2.5" + resolved "https://registry.yarnpkg.com/handle-thing/-/handle-thing-1.2.5.tgz#fd7aad726bf1a5fd16dfc29b2f7a6601d27139c4" + +handlebars@^4.0.3: + version "4.1.2" + resolved "https://registry.yarnpkg.com/handlebars/-/handlebars-4.1.2.tgz#b6b37c1ced0306b221e094fc7aca3ec23b131b67" + integrity sha512-nvfrjqvt9xQ8Z/w0ijewdD/vvWDTOweBUm96NTr66Wfvo1mJenBLwcYmPs3TIBP5ruzYGD7Hx/DaM9RmhroGPw== + dependencies: + neo-async "^2.6.0" + optimist "^0.6.1" + source-map "^0.6.1" + optionalDependencies: + uglify-js "^3.1.4" + +har-schema@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/har-schema/-/har-schema-1.0.5.tgz#d263135f43307c02c602afc8fe95970c0151369e" + +har-validator@~4.2.1: + version "4.2.1" + resolved "https://registry.yarnpkg.com/har-validator/-/har-validator-4.2.1.tgz#33481d0f1bbff600dd203d75812a6a5fba002e2a" + dependencies: + ajv "^4.9.1" + har-schema "^1.0.5" + +harmony-reflect@^1.4.6: + version "1.5.1" + resolved "https://registry.yarnpkg.com/harmony-reflect/-/harmony-reflect-1.5.1.tgz#b54ca617b00cc8aef559bbb17b3d85431dc7e329" + +has-ansi@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/has-ansi/-/has-ansi-2.0.0.tgz#34f5049ce1ecdf2b0649af3ef24e45ed35416d91" + dependencies: + ansi-regex "^2.0.0" + +has-flag@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-1.0.0.tgz#9d9e793165ce017a00f00418c43f942a7b1d11fa" + +has-flag@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-2.0.0.tgz#e8207af1cc7b30d446cc70b734b5e8be18f88d51" + +has-unicode@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/has-unicode/-/has-unicode-2.0.1.tgz#e0e6fe6a28cf51138855e086d1691e771de2a8b9" + +has@^1.0.1, has@~1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/has/-/has-1.0.1.tgz#8461733f538b0837c9361e39a9ab9e9704dc2f28" + dependencies: + function-bind "^1.0.2" + +hash-base@^2.0.0: + version "2.0.2" + resolved "https://registry.yarnpkg.com/hash-base/-/hash-base-2.0.2.tgz#66ea1d856db4e8a5470cadf6fce23ae5244ef2e1" + dependencies: + inherits "^2.0.1" + +hash-base@^3.0.0: + version "3.0.4" + resolved "https://registry.yarnpkg.com/hash-base/-/hash-base-3.0.4.tgz#5fc8686847ecd73499403319a6b0a3f3f6ae4918" + dependencies: + inherits "^2.0.1" + safe-buffer "^5.0.1" + +hash.js@^1.0.0, hash.js@^1.0.3: + version "1.1.3" + resolved "https://registry.yarnpkg.com/hash.js/-/hash.js-1.1.3.tgz#340dedbe6290187151c1ea1d777a3448935df846" + dependencies: + inherits "^2.0.3" + minimalistic-assert "^1.0.0" + +hawk@~3.1.3: + version "3.1.3" + resolved "https://registry.yarnpkg.com/hawk/-/hawk-3.1.3.tgz#078444bd7c1640b0fe540d2c9b73d59678e8e1c4" + dependencies: + boom "2.x.x" + cryptiles "2.x.x" + hoek "2.x.x" + sntp "1.x.x" + +he@1.1.x: + version "1.1.1" + resolved "https://registry.yarnpkg.com/he/-/he-1.1.1.tgz#93410fd21b009735151f8868c2f271f3427e23fd" + +history@^4.7.2: + version "4.7.2" + resolved "https://registry.yarnpkg.com/history/-/history-4.7.2.tgz#22b5c7f31633c5b8021c7f4a8a954ac139ee8d5b" + dependencies: + invariant "^2.2.1" + loose-envify "^1.2.0" + resolve-pathname "^2.2.0" + value-equal "^0.4.0" + warning "^3.0.0" + +hmac-drbg@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/hmac-drbg/-/hmac-drbg-1.0.1.tgz#d2745701025a6c775a6c545793ed502fc0c649a1" + dependencies: + hash.js "^1.0.3" + minimalistic-assert "^1.0.0" + minimalistic-crypto-utils "^1.0.1" + +hoek@2.x.x: + version "2.16.3" + resolved "https://registry.yarnpkg.com/hoek/-/hoek-2.16.3.tgz#20bb7403d3cea398e91dc4710a8ff1b8274a25ed" + +hoist-non-react-statics@^2.3.0: + version "2.3.1" + resolved "https://registry.yarnpkg.com/hoist-non-react-statics/-/hoist-non-react-statics-2.3.1.tgz#343db84c6018c650778898240135a1420ee22ce0" + +home-or-tmp@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/home-or-tmp/-/home-or-tmp-2.0.0.tgz#e36c3f2d2cae7d746a857e38d18d5f32a7882db8" + dependencies: + os-homedir "^1.0.0" + os-tmpdir "^1.0.1" + +hosted-git-info@^2.1.4: + version "2.5.0" + resolved "https://registry.yarnpkg.com/hosted-git-info/-/hosted-git-info-2.5.0.tgz#6d60e34b3abbc8313062c3b798ef8d901a07af3c" + +hpack.js@^2.1.6: + version "2.1.6" + resolved "https://registry.yarnpkg.com/hpack.js/-/hpack.js-2.1.6.tgz#87774c0949e513f42e84575b3c45681fade2a0b2" + dependencies: + inherits "^2.0.1" + obuf "^1.0.0" + readable-stream "^2.0.1" + wbuf "^1.1.0" + +html-comment-regex@^1.1.0: + version "1.1.1" + resolved "https://registry.yarnpkg.com/html-comment-regex/-/html-comment-regex-1.1.1.tgz#668b93776eaae55ebde8f3ad464b307a4963625e" + +html-encoding-sniffer@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/html-encoding-sniffer/-/html-encoding-sniffer-1.0.1.tgz#79bf7a785ea495fe66165e734153f363ff5437da" + dependencies: + whatwg-encoding "^1.0.1" + +html-entities@^1.2.0: + version "1.2.1" + resolved "https://registry.yarnpkg.com/html-entities/-/html-entities-1.2.1.tgz#0df29351f0721163515dfb9e5543e5f6eed5162f" + +html-minifier@^3.2.3: + version "3.5.3" + resolved "https://registry.yarnpkg.com/html-minifier/-/html-minifier-3.5.3.tgz#4a275e3b1a16639abb79b4c11191ff0d0fcf1ab9" + dependencies: + camel-case "3.0.x" + clean-css "4.1.x" + commander "2.11.x" + he "1.1.x" + ncname "1.0.x" + param-case "2.1.x" + relateurl "0.2.x" + uglify-js "3.0.x" + +html-webpack-plugin@^2.30.1: + version "2.30.1" + resolved "https://registry.yarnpkg.com/html-webpack-plugin/-/html-webpack-plugin-2.30.1.tgz#7f9c421b7ea91ec460f56527d78df484ee7537d5" + dependencies: + bluebird "^3.4.7" + html-minifier "^3.2.3" + loader-utils "^0.2.16" + lodash "^4.17.3" + pretty-error "^2.0.2" + toposort "^1.0.0" + +htmlparser2@^3.9.1: + version "3.9.2" + resolved "https://registry.yarnpkg.com/htmlparser2/-/htmlparser2-3.9.2.tgz#1bdf87acca0f3f9e53fa4fcceb0f4b4cbb00b338" + dependencies: + domelementtype "^1.3.0" + domhandler "^2.3.0" + domutils "^1.5.1" + entities "^1.1.1" + inherits "^2.0.1" + readable-stream "^2.0.2" + +htmlparser2@~3.3.0: + version "3.3.0" + resolved "https://registry.yarnpkg.com/htmlparser2/-/htmlparser2-3.3.0.tgz#cc70d05a59f6542e43f0e685c982e14c924a9efe" + dependencies: + domelementtype "1" + domhandler "2.1" + domutils "1.1" + readable-stream "1.0" + +http-deceiver@^1.2.7: + version "1.2.7" + resolved "https://registry.yarnpkg.com/http-deceiver/-/http-deceiver-1.2.7.tgz#fa7168944ab9a519d337cb0bec7284dc3e723d87" + +http-errors@~1.6.1, http-errors@~1.6.2: + version "1.6.2" + resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-1.6.2.tgz#0a002cc85707192a7e7946ceedc11155f60ec736" + dependencies: + depd "1.1.1" + inherits "2.0.3" + setprototypeof "1.0.3" + statuses ">= 1.3.1 < 2" + +http-proxy-middleware@~0.17.4: + version "0.17.4" + resolved "https://registry.yarnpkg.com/http-proxy-middleware/-/http-proxy-middleware-0.17.4.tgz#642e8848851d66f09d4f124912846dbaeb41b833" + dependencies: + http-proxy "^1.16.2" + is-glob "^3.1.0" + lodash "^4.17.2" + micromatch "^2.3.11" + +http-proxy@^1.16.2: + version "1.16.2" + resolved "https://registry.yarnpkg.com/http-proxy/-/http-proxy-1.16.2.tgz#06dff292952bf64dbe8471fa9df73066d4f37742" + dependencies: + eventemitter3 "1.x.x" + requires-port "1.x.x" + +http-signature@~1.1.0: + version "1.1.1" + resolved "https://registry.yarnpkg.com/http-signature/-/http-signature-1.1.1.tgz#df72e267066cd0ac67fb76adf8e134a8fbcf91bf" + dependencies: + assert-plus "^0.2.0" + jsprim "^1.2.2" + sshpk "^1.7.0" + +https-browserify@0.0.1: + version "0.0.1" + resolved "https://registry.yarnpkg.com/https-browserify/-/https-browserify-0.0.1.tgz#3f91365cabe60b77ed0ebba24b454e3e09d95a82" + +humanize-duration@^3.10.1: + version "3.10.1" + resolved "https://registry.yarnpkg.com/humanize-duration/-/humanize-duration-3.10.1.tgz#65b550c0aa095156ecb7c340db44ee0bdf71af4b" + +iconv-lite@0.4.13: + version "0.4.13" + resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.13.tgz#1f88aba4ab0b1508e8312acc39345f36e992e2f2" + +iconv-lite@^0.4.17, iconv-lite@~0.4.13: + version "0.4.19" + resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.19.tgz#f7468f60135f5e5dad3399c0a81be9a1603a082b" + +icss-replace-symbols@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/icss-replace-symbols/-/icss-replace-symbols-1.1.0.tgz#06ea6f83679a7749e386cfe1fe812ae5db223ded" + +icss-utils@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/icss-utils/-/icss-utils-2.1.0.tgz#83f0a0ec378bf3246178b6c2ad9136f135b1c962" + dependencies: + postcss "^6.0.1" + +identity-obj-proxy@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/identity-obj-proxy/-/identity-obj-proxy-3.0.0.tgz#94d2bda96084453ef36fbc5aaec37e0f79f1fc14" + dependencies: + harmony-reflect "^1.4.6" + +ieee754@^1.1.4: + version "1.1.8" + resolved "https://registry.yarnpkg.com/ieee754/-/ieee754-1.1.8.tgz#be33d40ac10ef1926701f6f08a2d86fbfd1ad3e4" + +ignore@^3.3.3: + version "3.3.5" + resolved "https://registry.yarnpkg.com/ignore/-/ignore-3.3.5.tgz#c4e715455f6073a8d7e5dae72d2fc9d71663dba6" + +image-size@~0.5.0: + version "0.5.5" + resolved "https://registry.yarnpkg.com/image-size/-/image-size-0.5.5.tgz#09dfd4ab9d20e29eb1c3e80b8990378df9e3cb9c" + +immutability-helper@^2.1.2: + version "2.3.1" + resolved "https://registry.yarnpkg.com/immutability-helper/-/immutability-helper-2.3.1.tgz#8ccfce92157208c120b2afad7ed05c11114c086e" + dependencies: + invariant "^2.2.0" + +imurmurhash@^0.1.4: + version "0.1.4" + resolved "https://registry.yarnpkg.com/imurmurhash/-/imurmurhash-0.1.4.tgz#9218b9b2b928a238b13dc4fb6b6d576f231453ea" + +indent-string@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/indent-string/-/indent-string-2.1.0.tgz#8e2d48348742121b4a8218b7a137e9a52049dc80" + dependencies: + repeating "^2.0.0" + +indexes-of@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/indexes-of/-/indexes-of-1.0.1.tgz#f30f716c8e2bd346c7b67d3df3915566a7c05607" + +indexof@0.0.1: + version "0.0.1" + resolved "https://registry.yarnpkg.com/indexof/-/indexof-0.0.1.tgz#82dc336d232b9062179d05ab3293a66059fd435d" + +inflight@^1.0.4: + version "1.0.6" + resolved "https://registry.yarnpkg.com/inflight/-/inflight-1.0.6.tgz#49bd6331d7d02d0c09bc910a1075ba8165b56df9" + dependencies: + once "^1.3.0" + wrappy "1" + +inherits@2, inherits@2.0.3, inherits@^2.0.1, inherits@^2.0.3, inherits@~2.0.0, inherits@~2.0.1, inherits@~2.0.3: + version "2.0.3" + resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.3.tgz#633c2c83e3da42a502f52466022480f4208261de" + +inherits@2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.1.tgz#b17d08d326b4423e568eff719f91b0b1cbdf69f1" + +ini@~1.3.0: + version "1.3.4" + resolved "https://registry.yarnpkg.com/ini/-/ini-1.3.4.tgz#0537cb79daf59b59a1a517dff706c86ec039162e" + +inquirer@^3.0.6: + version "3.2.3" + resolved "https://registry.yarnpkg.com/inquirer/-/inquirer-3.2.3.tgz#1c7b1731cf77b934ec47d22c9ac5aa8fe7fbe095" + dependencies: + ansi-escapes "^2.0.0" + chalk "^2.0.0" + cli-cursor "^2.1.0" + cli-width "^2.0.0" + external-editor "^2.0.4" + figures "^2.0.0" + lodash "^4.3.0" + mute-stream "0.0.7" + run-async "^2.2.0" + rx-lite "^4.0.8" + rx-lite-aggregates "^4.0.8" + string-width "^2.1.0" + strip-ansi "^4.0.0" + through "^2.3.6" + +internal-ip@^1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/internal-ip/-/internal-ip-1.2.0.tgz#ae9fbf93b984878785d50a8de1b356956058cf5c" + dependencies: + meow "^3.3.0" + +interpret@^1.0.0: + version "1.0.3" + resolved "https://registry.yarnpkg.com/interpret/-/interpret-1.0.3.tgz#cbc35c62eeee73f19ab7b10a801511401afc0f90" + +invariant@^2.2.0, invariant@^2.2.1, invariant@^2.2.2: + version "2.2.2" + resolved "https://registry.yarnpkg.com/invariant/-/invariant-2.2.2.tgz#9e1f56ac0acdb6bf303306f338be3b204ae60360" + dependencies: + loose-envify "^1.0.0" + +invert-kv@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/invert-kv/-/invert-kv-1.0.0.tgz#104a8e4aaca6d3d8cd157a8ef8bfab2d7a3ffdb6" + +ip@^1.1.0, ip@^1.1.5: + version "1.1.5" + resolved "https://registry.yarnpkg.com/ip/-/ip-1.1.5.tgz#bdded70114290828c0a039e72ef25f5aaec4354a" + +ipaddr.js@1.4.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/ipaddr.js/-/ipaddr.js-1.4.0.tgz#296aca878a821816e5b85d0a285a99bcff4582f0" + +is-absolute-url@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/is-absolute-url/-/is-absolute-url-2.1.0.tgz#50530dfb84fcc9aa7dbe7852e83a37b93b9f2aa6" + +is-arrayish@^0.2.1: + version "0.2.1" + resolved "https://registry.yarnpkg.com/is-arrayish/-/is-arrayish-0.2.1.tgz#77c99840527aa8ecb1a8ba697b80645a7a926a9d" + +is-binary-path@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/is-binary-path/-/is-binary-path-1.0.1.tgz#75f16642b480f187a711c814161fd3a4a7655898" + dependencies: + binary-extensions "^1.0.0" + +is-buffer@^1.1.5: + version "1.1.5" + resolved "https://registry.yarnpkg.com/is-buffer/-/is-buffer-1.1.5.tgz#1f3b26ef613b214b88cbca23cc6c01d87961eecc" + +is-builtin-module@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/is-builtin-module/-/is-builtin-module-1.0.0.tgz#540572d34f7ac3119f8f76c30cbc1b1e037affbe" + dependencies: + builtin-modules "^1.0.0" + +is-callable@^1.1.1, is-callable@^1.1.3: + version "1.1.3" + resolved "https://registry.yarnpkg.com/is-callable/-/is-callable-1.1.3.tgz#86eb75392805ddc33af71c92a0eedf74ee7604b2" + +is-ci@^1.0.10: + version "1.0.10" + resolved "https://registry.yarnpkg.com/is-ci/-/is-ci-1.0.10.tgz#f739336b2632365061a9d48270cd56ae3369318e" + dependencies: + ci-info "^1.0.0" + +is-date-object@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/is-date-object/-/is-date-object-1.0.1.tgz#9aa20eb6aeebbff77fbd33e74ca01b33581d3a16" + +is-dotfile@^1.0.0: + version "1.0.3" + resolved "https://registry.yarnpkg.com/is-dotfile/-/is-dotfile-1.0.3.tgz#a6a2f32ffd2dfb04f5ca25ecd0f6b83cf798a1e1" + +is-equal-shallow@^0.1.3: + version "0.1.3" + resolved "https://registry.yarnpkg.com/is-equal-shallow/-/is-equal-shallow-0.1.3.tgz#2238098fc221de0bcfa5d9eac4c45d638aa1c534" + dependencies: + is-primitive "^2.0.0" + +is-extendable@^0.1.1: + version "0.1.1" + resolved "https://registry.yarnpkg.com/is-extendable/-/is-extendable-0.1.1.tgz#62b110e289a471418e3ec36a617d472e301dfc89" + +is-extglob@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/is-extglob/-/is-extglob-1.0.0.tgz#ac468177c4943405a092fc8f29760c6ffc6206c0" + +is-extglob@^2.1.0: + version "2.1.1" + resolved "https://registry.yarnpkg.com/is-extglob/-/is-extglob-2.1.1.tgz#a88c02535791f02ed37c76a1b9ea9773c833f8c2" + +is-finite@^1.0.0: + version "1.0.2" + resolved "https://registry.yarnpkg.com/is-finite/-/is-finite-1.0.2.tgz#cc6677695602be550ef11e8b4aa6305342b6d0aa" + dependencies: + number-is-nan "^1.0.0" + +is-fullwidth-code-point@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz#ef9e31386f031a7f0d643af82fde50c457ef00cb" + dependencies: + number-is-nan "^1.0.0" + +is-fullwidth-code-point@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz#a3b30a5c4f199183167aaab93beefae3ddfb654f" + +is-function@~1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/is-function/-/is-function-1.0.1.tgz#12cfb98b65b57dd3d193a3121f5f6e2f437602b5" + +is-glob@^2.0.0, is-glob@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/is-glob/-/is-glob-2.0.1.tgz#d096f926a3ded5600f3fdfd91198cb0888c2d863" + dependencies: + is-extglob "^1.0.0" + +is-glob@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/is-glob/-/is-glob-3.1.0.tgz#7ba5ae24217804ac70707b96922567486cc3e84a" + dependencies: + is-extglob "^2.1.0" + +is-number@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/is-number/-/is-number-2.1.0.tgz#01fcbbb393463a548f2f466cce16dece49db908f" + dependencies: + kind-of "^3.0.2" + +is-number@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/is-number/-/is-number-3.0.0.tgz#24fd6201a4782cf50561c810276afc7d12d71195" + dependencies: + kind-of "^3.0.2" + +is-path-cwd@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/is-path-cwd/-/is-path-cwd-1.0.0.tgz#d225ec23132e89edd38fda767472e62e65f1106d" + +is-path-in-cwd@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/is-path-in-cwd/-/is-path-in-cwd-1.0.0.tgz#6477582b8214d602346094567003be8a9eac04dc" + dependencies: + is-path-inside "^1.0.0" + +is-path-inside@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/is-path-inside/-/is-path-inside-1.0.0.tgz#fc06e5a1683fbda13de667aff717bbc10a48f37f" + dependencies: + path-is-inside "^1.0.1" + +is-plain-obj@^1.0.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/is-plain-obj/-/is-plain-obj-1.1.0.tgz#71a50c8429dfca773c92a390a4a03b39fcd51d3e" + +is-posix-bracket@^0.1.0: + version "0.1.1" + resolved "https://registry.yarnpkg.com/is-posix-bracket/-/is-posix-bracket-0.1.1.tgz#3334dc79774368e92f016e6fbc0a88f5cd6e6bc4" + +is-primitive@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/is-primitive/-/is-primitive-2.0.0.tgz#207bab91638499c07b2adf240a41a87210034575" + +is-promise@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/is-promise/-/is-promise-2.1.0.tgz#79a2a9ece7f096e80f36d2b2f3bc16c1ff4bf3fa" + +is-regex@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/is-regex/-/is-regex-1.0.4.tgz#5517489b547091b0930e095654ced25ee97e9491" + dependencies: + has "^1.0.1" + +is-resolvable@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/is-resolvable/-/is-resolvable-1.0.0.tgz#8df57c61ea2e3c501408d100fb013cf8d6e0cc62" + dependencies: + tryit "^1.0.1" + +is-stream@^1.0.1, is-stream@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/is-stream/-/is-stream-1.1.0.tgz#12d4a3dd4e68e0b79ceb8dbc84173ae80d91ca44" + +is-subset@^0.1.1: + version "0.1.1" + resolved "https://registry.yarnpkg.com/is-subset/-/is-subset-0.1.1.tgz#8a59117d932de1de00f245fcdd39ce43f1e939a6" + +is-svg@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/is-svg/-/is-svg-2.1.0.tgz#cf61090da0d9efbcab8722deba6f032208dbb0e9" + dependencies: + html-comment-regex "^1.1.0" + +is-symbol@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/is-symbol/-/is-symbol-1.0.1.tgz#3cc59f00025194b6ab2e38dbae6689256b660572" + +is-typedarray@~1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/is-typedarray/-/is-typedarray-1.0.0.tgz#e479c80858df0c1b11ddda6940f96011fcda4a9a" + +is-utf8@^0.2.0: + version "0.2.1" + resolved "https://registry.yarnpkg.com/is-utf8/-/is-utf8-0.2.1.tgz#4b0da1442104d1b336340e80797e865cf39f7d72" + +is-windows@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/is-windows/-/is-windows-1.0.1.tgz#310db70f742d259a16a369202b51af84233310d9" + +isarray@0.0.1: + version "0.0.1" + resolved "https://registry.yarnpkg.com/isarray/-/isarray-0.0.1.tgz#8a18acfca9a8f4177e09abfc6038939b05d1eedf" + +isarray@1.0.0, isarray@^1.0.0, isarray@~1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/isarray/-/isarray-1.0.0.tgz#bb935d48582cba168c06834957a54a3e07124f11" + +isexe@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10" + +isobject@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/isobject/-/isobject-2.1.0.tgz#f065561096a3f1da2ef46272f815c840d87e0c89" + dependencies: + isarray "1.0.0" + +isomorphic-fetch@^2.1.1: + version "2.2.1" + resolved "https://registry.yarnpkg.com/isomorphic-fetch/-/isomorphic-fetch-2.2.1.tgz#611ae1acf14f5e81f729507472819fe9733558a9" + dependencies: + node-fetch "^1.0.1" + whatwg-fetch ">=0.10.0" + +isstream@~0.1.2: + version "0.1.2" + resolved "https://registry.yarnpkg.com/isstream/-/isstream-0.1.2.tgz#47e63f7af55afa6f92e1500e690eb8b8529c099a" + +istanbul-api@^1.1.1: + version "1.1.14" + resolved "https://registry.yarnpkg.com/istanbul-api/-/istanbul-api-1.1.14.tgz#25bc5701f7c680c0ffff913de46e3619a3a6e680" + dependencies: + async "^2.1.4" + fileset "^2.0.2" + istanbul-lib-coverage "^1.1.1" + istanbul-lib-hook "^1.0.7" + istanbul-lib-instrument "^1.8.0" + istanbul-lib-report "^1.1.1" + istanbul-lib-source-maps "^1.2.1" + istanbul-reports "^1.1.2" + js-yaml "^3.7.0" + mkdirp "^0.5.1" + once "^1.4.0" + +istanbul-lib-coverage@^1.0.1, istanbul-lib-coverage@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/istanbul-lib-coverage/-/istanbul-lib-coverage-1.1.1.tgz#73bfb998885299415c93d38a3e9adf784a77a9da" + +istanbul-lib-hook@^1.0.7: + version "1.0.7" + resolved "https://registry.yarnpkg.com/istanbul-lib-hook/-/istanbul-lib-hook-1.0.7.tgz#dd6607f03076578fe7d6f2a630cf143b49bacddc" + dependencies: + append-transform "^0.4.0" + +istanbul-lib-instrument@^1.4.2, istanbul-lib-instrument@^1.7.2, istanbul-lib-instrument@^1.8.0: + version "1.8.0" + resolved "https://registry.yarnpkg.com/istanbul-lib-instrument/-/istanbul-lib-instrument-1.8.0.tgz#66f6c9421cc9ec4704f76f2db084ba9078a2b532" + dependencies: + babel-generator "^6.18.0" + babel-template "^6.16.0" + babel-traverse "^6.18.0" + babel-types "^6.18.0" + babylon "^6.18.0" + istanbul-lib-coverage "^1.1.1" + semver "^5.3.0" + +istanbul-lib-report@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/istanbul-lib-report/-/istanbul-lib-report-1.1.1.tgz#f0e55f56655ffa34222080b7a0cd4760e1405fc9" + dependencies: + istanbul-lib-coverage "^1.1.1" + mkdirp "^0.5.1" + path-parse "^1.0.5" + supports-color "^3.1.2" + +istanbul-lib-source-maps@^1.1.0, istanbul-lib-source-maps@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/istanbul-lib-source-maps/-/istanbul-lib-source-maps-1.2.1.tgz#a6fe1acba8ce08eebc638e572e294d267008aa0c" + dependencies: + debug "^2.6.3" + istanbul-lib-coverage "^1.1.1" + mkdirp "^0.5.1" + rimraf "^2.6.1" + source-map "^0.5.3" + +istanbul-reports@^1.1.2: + version "1.1.2" + resolved "https://registry.yarnpkg.com/istanbul-reports/-/istanbul-reports-1.1.2.tgz#0fb2e3f6aa9922bd3ce45d05d8ab4d5e8e07bd4f" + dependencies: + handlebars "^4.0.3" + +jasmine-expect: + version "3.7.1" + resolved "https://registry.yarnpkg.com/jasmine-expect/-/jasmine-expect-3.7.1.tgz#37037bedee2e83412273573f0a4305973c7e71e5" + dependencies: + add-matchers "0.5.0" + +jest-changed-files@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-changed-files/-/jest-changed-files-21.0.2.tgz#0a74f35cf2d3b7c8ef9ab4fac0a75409f81ec1b0" + dependencies: + throat "^4.0.0" + +jest-cli@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-cli/-/jest-cli-21.0.2.tgz#2e08af63d44fc21284ebf496cf71e381f3cc9786" + dependencies: + ansi-escapes "^3.0.0" + chalk "^2.0.1" + glob "^7.1.2" + graceful-fs "^4.1.11" + is-ci "^1.0.10" + istanbul-api "^1.1.1" + istanbul-lib-coverage "^1.0.1" + istanbul-lib-instrument "^1.4.2" + istanbul-lib-source-maps "^1.1.0" + jest-changed-files "^21.0.2" + jest-config "^21.0.2" + jest-environment-jsdom "^21.0.2" + jest-haste-map "^21.0.2" + jest-message-util "^21.0.2" + jest-regex-util "^21.0.2" + jest-resolve-dependencies "^21.0.2" + jest-runner "^21.0.2" + jest-runtime "^21.0.2" + jest-snapshot "^21.0.2" + jest-util "^21.0.2" + micromatch "^2.3.11" + node-notifier "^5.0.2" + pify "^3.0.0" + slash "^1.0.0" + string-length "^2.0.0" + strip-ansi "^4.0.0" + which "^1.2.12" + worker-farm "^1.3.1" + yargs "^9.0.0" + +jest-config@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-config/-/jest-config-21.0.2.tgz#ea42b94f3c22ae4e4aa11c69f5b45e34e342199d" + dependencies: + chalk "^2.0.1" + glob "^7.1.1" + jest-environment-jsdom "^21.0.2" + jest-environment-node "^21.0.2" + jest-get-type "^21.0.2" + jest-jasmine2 "^21.0.2" + jest-regex-util "^21.0.2" + jest-resolve "^21.0.2" + jest-util "^21.0.2" + jest-validate "^21.0.2" + pretty-format "^21.0.2" + +jest-diff@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-diff/-/jest-diff-21.0.2.tgz#751014f36ad5d505f6affce5542fde0e444ee50a" + dependencies: + chalk "^2.0.1" + diff "^3.2.0" + jest-get-type "^21.0.2" + pretty-format "^21.0.2" + +jest-docblock@^20.0.1: + version "20.0.3" + resolved "https://registry.yarnpkg.com/jest-docblock/-/jest-docblock-20.0.3.tgz#17bea984342cc33d83c50fbe1545ea0efaa44712" + +jest-docblock@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-docblock/-/jest-docblock-21.0.2.tgz#66f69ddb440799fc32f91d0ac3d8d35e99e2032f" + +jest-environment-jsdom@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-environment-jsdom/-/jest-environment-jsdom-21.0.2.tgz#6f6ab5bd71970d1900fbd47a46701c0a07fa3be5" + dependencies: + jest-mock "^21.0.2" + jest-util "^21.0.2" + jsdom "^9.12.0" + +jest-environment-node@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-environment-node/-/jest-environment-node-21.0.2.tgz#4267ceb39551f8ecaed182ab882a93ef4d5de240" + dependencies: + jest-mock "^21.0.2" + jest-util "^21.0.2" + +jest-get-type@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-get-type/-/jest-get-type-21.0.2.tgz#304e6b816dd33cd1f47aba0597bcad258a509fc6" + +jest-haste-map@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-haste-map/-/jest-haste-map-21.0.2.tgz#bd98bc6cd6f207eb029b2f5918da1a9347eb11b7" + dependencies: + fb-watchman "^2.0.0" + graceful-fs "^4.1.11" + jest-docblock "^21.0.2" + micromatch "^2.3.11" + sane "^2.0.0" + worker-farm "^1.3.1" + +jest-jasmine2@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-jasmine2/-/jest-jasmine2-21.0.2.tgz#a368abb3a686def4d6e763509a265104943cd469" + dependencies: + chalk "^2.0.1" + expect "^21.0.2" + graceful-fs "^4.1.11" + jest-diff "^21.0.2" + jest-matcher-utils "^21.0.2" + jest-message-util "^21.0.2" + jest-snapshot "^21.0.2" + p-cancelable "^0.3.0" + +jest-matcher-utils@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-matcher-utils/-/jest-matcher-utils-21.0.2.tgz#eb6736a45b698546d71f7e1ffbbd36587eeb27bc" + dependencies: + chalk "^2.0.1" + jest-get-type "^21.0.2" + pretty-format "^21.0.2" + +jest-message-util@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-message-util/-/jest-message-util-21.0.2.tgz#81242e07d426ad54c15f3d7c55b072e9db7b39a9" + dependencies: + chalk "^2.0.1" + micromatch "^2.3.11" + slash "^1.0.0" + +jest-mock@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-mock/-/jest-mock-21.0.2.tgz#5e92902450e1ce78be3864cc4d50dbd5d1582fbd" + +jest-regex-util@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-regex-util/-/jest-regex-util-21.0.2.tgz#06248c07b53ff444223ebe8e33a25bc051ac976f" + +jest-resolve-dependencies@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-resolve-dependencies/-/jest-resolve-dependencies-21.0.2.tgz#c42cc371034023ac1a226a7a52f86233c8871938" + dependencies: + jest-regex-util "^21.0.2" + +jest-resolve@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-resolve/-/jest-resolve-21.0.2.tgz#57b2c20cbeca2357eb5e638d5c28beca7f38c3f8" + dependencies: + browser-resolve "^1.11.2" + chalk "^2.0.1" + is-builtin-module "^1.0.0" + +jest-runner@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-runner/-/jest-runner-21.0.2.tgz#1462d431d25f7744e8b5e03837bbf9e268dc8b15" + dependencies: + jest-config "^21.0.2" + jest-docblock "^21.0.2" + jest-haste-map "^21.0.2" + jest-jasmine2 "^21.0.2" + jest-message-util "^21.0.2" + jest-runtime "^21.0.2" + jest-util "^21.0.2" + pify "^3.0.0" + throat "^4.0.0" + worker-farm "^1.3.1" + +jest-runtime@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-runtime/-/jest-runtime-21.0.2.tgz#ce26ba06bcd5501991bd994b1eacc0c7c7913895" + dependencies: + babel-core "^6.0.0" + babel-jest "^21.0.2" + babel-plugin-istanbul "^4.0.0" + chalk "^2.0.1" + convert-source-map "^1.4.0" + graceful-fs "^4.1.11" + jest-config "^21.0.2" + jest-haste-map "^21.0.2" + jest-regex-util "^21.0.2" + jest-resolve "^21.0.2" + jest-util "^21.0.2" + json-stable-stringify "^1.0.1" + micromatch "^2.3.11" + slash "^1.0.0" + strip-bom "3.0.0" + write-file-atomic "^2.1.0" + yargs "^9.0.0" + +jest-snapshot@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-snapshot/-/jest-snapshot-21.0.2.tgz#5b8f4dd05c1759381db835451fba4bcd85a55611" + dependencies: + chalk "^2.0.1" + jest-diff "^21.0.2" + jest-matcher-utils "^21.0.2" + mkdirp "^0.5.1" + natural-compare "^1.4.0" + pretty-format "^21.0.2" + +jest-util@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-util/-/jest-util-21.0.2.tgz#3ee2380af25c414a39f07b23c84da6f2d5f1f76a" + dependencies: + callsites "^2.0.0" + chalk "^2.0.1" + graceful-fs "^4.1.11" + jest-message-util "^21.0.2" + jest-mock "^21.0.2" + jest-validate "^21.0.2" + mkdirp "^0.5.1" + +jest-validate@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest-validate/-/jest-validate-21.0.2.tgz#dd066b257bd102759c214747d73bed6bcfa4349d" + dependencies: + chalk "^2.0.1" + jest-get-type "^21.0.2" + leven "^2.1.0" + pretty-format "^21.0.2" + +jest@^21.0.1: + version "21.0.2" + resolved "https://registry.yarnpkg.com/jest/-/jest-21.0.2.tgz#a5c9bdc9d4322ae672fe8cb3eaf25c268c5f04b2" + dependencies: + jest-cli "^21.0.2" + +js-base64@^2.1.9: + version "2.2.0" + resolved "https://registry.yarnpkg.com/js-base64/-/js-base64-2.2.0.tgz#5e8a8d193a908198dd23d1704826d207b0e5a8f6" + +js-tokens@^3.0.0, js-tokens@^3.0.2: + version "3.0.2" + resolved "https://registry.yarnpkg.com/js-tokens/-/js-tokens-3.0.2.tgz#9866df395102130e38f7f996bceb65443209c25b" + +js-yaml@^3.7.0, js-yaml@^3.9.1: + version "3.9.1" + resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-3.9.1.tgz#08775cebdfdd359209f0d2acd383c8f86a6904a0" + dependencies: + argparse "^1.0.7" + esprima "^4.0.0" + +js-yaml@~3.7.0: + version "3.7.0" + resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-3.7.0.tgz#5c967ddd837a9bfdca5f2de84253abe8a1c03b80" + dependencies: + argparse "^1.0.7" + esprima "^2.6.0" + +js2xmlparser@~3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/js2xmlparser/-/js2xmlparser-3.0.0.tgz#3fb60eaa089c5440f9319f51760ccd07e2499733" + dependencies: + xmlcreate "^1.0.1" + +jsbn@~0.1.0: + version "0.1.1" + resolved "https://registry.yarnpkg.com/jsbn/-/jsbn-0.1.1.tgz#a5e654c2e5a2deb5f201d96cefbca80c0ef2f513" + +jschardet@^1.4.2: + version "1.5.1" + resolved "https://registry.yarnpkg.com/jschardet/-/jschardet-1.5.1.tgz#c519f629f86b3a5bedba58a88d311309eec097f9" + +jsdoc@^3.5.4: + version "3.5.4" + resolved "https://registry.yarnpkg.com/jsdoc/-/jsdoc-3.5.4.tgz#ceeef7c4bac4335cb10ff41e3a0f58939a534428" + dependencies: + babylon "~7.0.0-beta.19" + bluebird "~3.5.0" + catharsis "~0.8.9" + escape-string-regexp "~1.0.5" + js2xmlparser "~3.0.0" + klaw "~2.0.0" + marked "~0.3.6" + mkdirp "~0.5.1" + requizzle "~0.2.1" + strip-json-comments "~2.0.1" + taffydb "2.6.2" + underscore "~1.8.3" + +jsdom@^9.12.0: + version "9.12.0" + resolved "https://registry.yarnpkg.com/jsdom/-/jsdom-9.12.0.tgz#e8c546fffcb06c00d4833ca84410fed7f8a097d4" + dependencies: + abab "^1.0.3" + acorn "^4.0.4" + acorn-globals "^3.1.0" + array-equal "^1.0.0" + content-type-parser "^1.0.1" + cssom ">= 0.3.2 < 0.4.0" + cssstyle ">= 0.2.37 < 0.3.0" + escodegen "^1.6.1" + html-encoding-sniffer "^1.0.1" + nwmatcher ">= 1.3.9 < 2.0.0" + parse5 "^1.5.1" + request "^2.79.0" + sax "^1.2.1" + symbol-tree "^3.2.1" + tough-cookie "^2.3.2" + webidl-conversions "^4.0.0" + whatwg-encoding "^1.0.1" + whatwg-url "^4.3.0" + xml-name-validator "^2.0.1" + +jsesc@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/jsesc/-/jsesc-1.3.0.tgz#46c3fec8c1892b12b0833db9bc7622176dbab34b" + +jsesc@~0.5.0: + version "0.5.0" + resolved "https://registry.yarnpkg.com/jsesc/-/jsesc-0.5.0.tgz#e7dee66e35d6fc16f710fe91d5cf69f70f08911d" + +json-loader@^0.5.4: + version "0.5.7" + resolved "https://registry.yarnpkg.com/json-loader/-/json-loader-0.5.7.tgz#dca14a70235ff82f0ac9a3abeb60d337a365185d" + +json-schema-traverse@^0.3.0: + version "0.3.1" + resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.3.1.tgz#349a6d44c53a51de89b40805c5d5e59b417d3340" + +json-schema@0.2.3: + version "0.2.3" + resolved "https://registry.yarnpkg.com/json-schema/-/json-schema-0.2.3.tgz#b480c892e59a2f05954ce727bd3f2a4e882f9e13" + +json-stable-stringify@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/json-stable-stringify/-/json-stable-stringify-1.0.1.tgz#9a759d39c5f2ff503fd5300646ed445f88c4f9af" + dependencies: + jsonify "~0.0.0" + +json-stringify-safe@~5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz#1296a2d58fd45f19a0f6ce01d65701e2c735b6eb" + +json3@^3.3.2: + version "3.3.2" + resolved "https://registry.yarnpkg.com/json3/-/json3-3.3.2.tgz#3c0434743df93e2f5c42aee7b19bcb483575f4e1" + +json5@^0.5.0, json5@^0.5.1: + version "0.5.1" + resolved "https://registry.yarnpkg.com/json5/-/json5-0.5.1.tgz#1eade7acc012034ad84e2396767ead9fa5495821" + +jsonify@~0.0.0: + version "0.0.0" + resolved "https://registry.yarnpkg.com/jsonify/-/jsonify-0.0.0.tgz#2c74b6ee41d93ca51b7b5aaee8f503631d252a73" + +jsprim@^1.2.2: + version "1.4.1" + resolved "https://registry.yarnpkg.com/jsprim/-/jsprim-1.4.1.tgz#313e66bc1e5cc06e438bc1b7499c2e5c56acb6a2" + dependencies: + assert-plus "1.0.0" + extsprintf "1.3.0" + json-schema "0.2.3" + verror "1.10.0" + +jsx-ast-utils@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/jsx-ast-utils/-/jsx-ast-utils-2.0.1.tgz#e801b1b39985e20fffc87b40e3748080e2dcac7f" + dependencies: + array-includes "^3.0.3" + +just-extend@^1.1.22: + version "1.1.22" + resolved "https://registry.yarnpkg.com/just-extend/-/just-extend-1.1.22.tgz#3330af756cab6a542700c64b2e4e4aa062d52fff" + +kind-of@^3.0.2: + version "3.2.2" + resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-3.2.2.tgz#31ea21a734bab9bbb0f32466d893aea51e4a3c64" + dependencies: + is-buffer "^1.1.5" + +kind-of@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-4.0.0.tgz#20813df3d712928b207378691a45066fae72dd57" + dependencies: + is-buffer "^1.1.5" + +klaw@~2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/klaw/-/klaw-2.0.0.tgz#59c128e0dc5ce410201151194eeb9cbf858650f6" + dependencies: + graceful-fs "^4.1.9" + +lazy-cache@^1.0.3: + version "1.0.4" + resolved "https://registry.yarnpkg.com/lazy-cache/-/lazy-cache-1.0.4.tgz#a1d78fc3a50474cb80845d3b3b6e1da49a446e8e" + +lcid@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/lcid/-/lcid-1.0.0.tgz#308accafa0bc483a3867b4b6f2b9506251d1b835" + dependencies: + invert-kv "^1.0.0" + +less-loader@^4.0.5: + version "4.0.5" + resolved "https://registry.yarnpkg.com/less-loader/-/less-loader-4.0.5.tgz#ae155a7406cac6acd293d785587fcff0f478c4dd" + dependencies: + clone "^2.1.1" + loader-utils "^1.1.0" + pify "^2.3.0" + +less@^2.7.2: + version "2.7.2" + resolved "https://registry.yarnpkg.com/less/-/less-2.7.2.tgz#368d6cc73e1fb03981183280918743c5dcf9b3df" + optionalDependencies: + errno "^0.1.1" + graceful-fs "^4.1.2" + image-size "~0.5.0" + mime "^1.2.11" + mkdirp "^0.5.0" + promise "^7.1.1" + request "^2.72.0" + source-map "^0.5.3" + +lesshint: + version "4.1.3" + resolved "https://registry.yarnpkg.com/lesshint/-/lesshint-4.1.3.tgz#7446407ac9c2546c45d171ede386fa4dbc075f59" + dependencies: + commander "^2.8.0" + lodash.merge "^4.0.1" + lodash.sortby "^4.0.1" + minimatch "^3.0.2" + postcss "^6.0.0" + postcss-less "^1.0.0" + postcss-selector-parser "^2.0.0" + postcss-values-parser "~1.2.2" + rcfinder "^0.1.8" + strip-json-comments "^2.0.0" + +leven@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/leven/-/leven-2.1.0.tgz#c2e7a9f772094dee9d34202ae8acce4687875580" + +levn@^0.3.0, levn@~0.3.0: + version "0.3.0" + resolved "https://registry.yarnpkg.com/levn/-/levn-0.3.0.tgz#3b09924edf9f083c0490fdd4c0bc4421e04764ee" + dependencies: + prelude-ls "~1.1.2" + type-check "~0.3.2" + +load-json-file@^1.0.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/load-json-file/-/load-json-file-1.1.0.tgz#956905708d58b4bab4c2261b04f59f31c99374c0" + dependencies: + graceful-fs "^4.1.2" + parse-json "^2.2.0" + pify "^2.0.0" + pinkie-promise "^2.0.0" + strip-bom "^2.0.0" + +load-json-file@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/load-json-file/-/load-json-file-2.0.0.tgz#7947e42149af80d696cbf797bcaabcfe1fe29ca8" + dependencies: + graceful-fs "^4.1.2" + parse-json "^2.2.0" + pify "^2.0.0" + strip-bom "^3.0.0" + +loader-runner@^2.3.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/loader-runner/-/loader-runner-2.3.0.tgz#f482aea82d543e07921700d5a46ef26fdac6b8a2" + +loader-utils@^0.2.16: + version "0.2.17" + resolved "https://registry.yarnpkg.com/loader-utils/-/loader-utils-0.2.17.tgz#f86e6374d43205a6e6c60e9196f17c0299bfb348" + dependencies: + big.js "^3.1.3" + emojis-list "^2.0.0" + json5 "^0.5.0" + object-assign "^4.0.1" + +loader-utils@^1.0.2, loader-utils@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/loader-utils/-/loader-utils-1.1.0.tgz#c98aef488bcceda2ffb5e2de646d6a754429f5cd" + dependencies: + big.js "^3.1.3" + emojis-list "^2.0.0" + json5 "^0.5.0" + +locate-path@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-2.0.0.tgz#2b568b265eec944c6d9c0de9c3dbbbca0354cd8e" + dependencies: + p-locate "^2.0.0" + path-exists "^3.0.0" + +lodash.assignin@^4.0.9: + version "4.2.0" + resolved "https://registry.yarnpkg.com/lodash.assignin/-/lodash.assignin-4.2.0.tgz#ba8df5fb841eb0a3e8044232b0e263a8dc6a28a2" + +lodash.bind@^4.1.4: + version "4.2.1" + resolved "https://registry.yarnpkg.com/lodash.bind/-/lodash.bind-4.2.1.tgz#7ae3017e939622ac31b7d7d7dcb1b34db1690d35" + +lodash.camelcase@^4.3.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/lodash.camelcase/-/lodash.camelcase-4.3.0.tgz#b28aa6288a2b9fc651035c7711f65ab6190331a6" + +lodash.clonedeep@^4.3.2: + version "4.5.0" + resolved "https://registry.yarnpkg.com/lodash.clonedeep/-/lodash.clonedeep-4.5.0.tgz#e23f3f9c4f8fbdde872529c1071857a086e5ccef" + +lodash.cond@^4.3.0: + version "4.5.2" + resolved "https://registry.yarnpkg.com/lodash.cond/-/lodash.cond-4.5.2.tgz#f471a1da486be60f6ab955d17115523dd1d255d5" + +lodash.defaults@^4.0.1: + version "4.2.0" + resolved "https://registry.yarnpkg.com/lodash.defaults/-/lodash.defaults-4.2.0.tgz#d09178716ffea4dde9e5fb7b37f6f0802274580c" + +lodash.filter@^4.4.0: + version "4.6.0" + resolved "https://registry.yarnpkg.com/lodash.filter/-/lodash.filter-4.6.0.tgz#668b1d4981603ae1cc5a6fa760143e480b4c4ace" + +lodash.flatten@^4.2.0: + version "4.4.0" + resolved "https://registry.yarnpkg.com/lodash.flatten/-/lodash.flatten-4.4.0.tgz#f31c22225a9632d2bbf8e4addbef240aa765a61f" + +lodash.foreach@^4.3.0: + version "4.5.0" + resolved "https://registry.yarnpkg.com/lodash.foreach/-/lodash.foreach-4.5.0.tgz#1a6a35eace401280c7f06dddec35165ab27e3e53" + +lodash.map@^4.4.0: + version "4.6.0" + resolved "https://registry.yarnpkg.com/lodash.map/-/lodash.map-4.6.0.tgz#771ec7839e3473d9c4cde28b19394c3562f4f6d3" + +lodash.memoize@^4.1.2: + version "4.1.2" + resolved "https://registry.yarnpkg.com/lodash.memoize/-/lodash.memoize-4.1.2.tgz#bcc6c49a42a2840ed997f323eada5ecd182e0bfe" + +lodash.merge@^4.0.1, lodash.merge@^4.4.0: + version "4.6.0" + resolved "https://registry.yarnpkg.com/lodash.merge/-/lodash.merge-4.6.0.tgz#69884ba144ac33fe699737a6086deffadd0f89c5" + +lodash.pick@^4.2.1: + version "4.4.0" + resolved "https://registry.yarnpkg.com/lodash.pick/-/lodash.pick-4.4.0.tgz#52f05610fff9ded422611441ed1fc123a03001b3" + +lodash.reduce@^4.4.0: + version "4.6.0" + resolved "https://registry.yarnpkg.com/lodash.reduce/-/lodash.reduce-4.6.0.tgz#f1ab6b839299ad48f784abbf476596f03b914d3b" + +lodash.reject@^4.4.0: + version "4.6.0" + resolved "https://registry.yarnpkg.com/lodash.reject/-/lodash.reject-4.6.0.tgz#80d6492dc1470864bbf583533b651f42a9f52415" + +lodash.some@^4.4.0: + version "4.6.0" + resolved "https://registry.yarnpkg.com/lodash.some/-/lodash.some-4.6.0.tgz#1bb9f314ef6b8baded13b549169b2a945eb68e4d" + +lodash.sortby@^4.0.1: + version "4.7.0" + resolved "https://registry.yarnpkg.com/lodash.sortby/-/lodash.sortby-4.7.0.tgz#edd14c824e2cc9c1e0b0a1b42bb5210516a42438" + +lodash.uniq@^4.5.0: + version "4.5.0" + resolved "https://registry.yarnpkg.com/lodash.uniq/-/lodash.uniq-4.5.0.tgz#d0225373aeb652adc1bc82e4945339a842754773" + +lodash@^4.0.0, lodash@^4.14.0, lodash@^4.17.2, lodash@^4.17.3, lodash@^4.17.4, lodash@^4.3.0: + version "4.17.15" + resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.15.tgz#b447f6670a0455bbfeedd11392eff330ea097548" + integrity sha512-8xOcRHvCjnocdS5cpwXQXVzmmh5e5+saE2QGoeQmbKmRS6J3VQppPOIt0MnmE+4xlZoumy0GPG0D0MVIQbNA1A== + +loglevel@^1.4.1: + version "1.4.1" + resolved "https://registry.yarnpkg.com/loglevel/-/loglevel-1.4.1.tgz#95b383f91a3c2756fd4ab093667e4309161f2bcd" + +lolex@^1.6.0: + version "1.6.0" + resolved "https://registry.yarnpkg.com/lolex/-/lolex-1.6.0.tgz#3a9a0283452a47d7439e72731b9e07d7386e49f6" + +lolex@^2.1.2: + version "2.1.2" + resolved "https://registry.yarnpkg.com/lolex/-/lolex-2.1.2.tgz#2694b953c9ea4d013e5b8bfba891c991025b2629" + +longest@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/longest/-/longest-1.0.1.tgz#30a0b2da38f73770e8294a0d22e6625ed77d0097" + +loose-envify@^1.0.0, loose-envify@^1.2.0, loose-envify@^1.3.1: + version "1.3.1" + resolved "https://registry.yarnpkg.com/loose-envify/-/loose-envify-1.3.1.tgz#d1a8ad33fa9ce0e713d65fdd0ac8b748d478c848" + dependencies: + js-tokens "^3.0.0" + +loud-rejection@^1.0.0: + version "1.6.0" + resolved "https://registry.yarnpkg.com/loud-rejection/-/loud-rejection-1.6.0.tgz#5b46f80147edee578870f086d04821cf998e551f" + dependencies: + currently-unhandled "^0.4.1" + signal-exit "^3.0.0" + +lower-case@^1.1.1: + version "1.1.4" + resolved "https://registry.yarnpkg.com/lower-case/-/lower-case-1.1.4.tgz#9a2cabd1b9e8e0ae993a4bf7d5875c39c42e8eac" + +lru-cache@^4.0.1: + version "4.1.1" + resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-4.1.1.tgz#622e32e82488b49279114a4f9ecf45e7cd6bba55" + dependencies: + pseudomap "^1.0.2" + yallist "^2.1.2" + +macaddress@^0.2.8: + version "0.2.8" + resolved "https://registry.yarnpkg.com/macaddress/-/macaddress-0.2.8.tgz#5904dc537c39ec6dbefeae902327135fa8511f12" + +make-dir@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/make-dir/-/make-dir-1.0.0.tgz#97a011751e91dd87cfadef58832ebb04936de978" + dependencies: + pify "^2.3.0" + +makeerror@1.0.x: + version "1.0.11" + resolved "https://registry.yarnpkg.com/makeerror/-/makeerror-1.0.11.tgz#e01a5c9109f2af79660e4e8b9587790184f5a96c" + dependencies: + tmpl "1.0.x" + +map-obj@^1.0.0, map-obj@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/map-obj/-/map-obj-1.0.1.tgz#d933ceb9205d82bdcf4886f6742bdc2b4dea146d" + +marked@~0.3.6: + version "0.3.6" + resolved "https://registry.yarnpkg.com/marked/-/marked-0.3.6.tgz#b2c6c618fccece4ef86c4fc6cb8a7cbf5aeda8d7" + +math-expression-evaluator@^1.2.14: + version "1.2.17" + resolved "https://registry.yarnpkg.com/math-expression-evaluator/-/math-expression-evaluator-1.2.17.tgz#de819fdbcd84dccd8fae59c6aeb79615b9d266ac" + +md5.js@^1.3.4: + version "1.3.4" + resolved "https://registry.yarnpkg.com/md5.js/-/md5.js-1.3.4.tgz#e9bdbde94a20a5ac18b04340fc5764d5b09d901d" + dependencies: + hash-base "^3.0.0" + inherits "^2.0.1" + +media-typer@0.3.0: + version "0.3.0" + resolved "https://registry.yarnpkg.com/media-typer/-/media-typer-0.3.0.tgz#8710d7af0aa626f8fffa1ce00168545263255748" + +mem@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/mem/-/mem-1.1.0.tgz#5edd52b485ca1d900fe64895505399a0dfa45f76" + dependencies: + mimic-fn "^1.0.0" + +memory-fs@^0.4.0, memory-fs@~0.4.1: + version "0.4.1" + resolved "https://registry.yarnpkg.com/memory-fs/-/memory-fs-0.4.1.tgz#3a9a20b8462523e447cfbc7e8bb80ed667bfc552" + dependencies: + errno "^0.1.3" + readable-stream "^2.0.1" + +meow@^3.3.0: + version "3.7.0" + resolved "https://registry.yarnpkg.com/meow/-/meow-3.7.0.tgz#72cb668b425228290abbfa856892587308a801fb" + dependencies: + camelcase-keys "^2.0.0" + decamelize "^1.1.2" + loud-rejection "^1.0.0" + map-obj "^1.0.1" + minimist "^1.1.3" + normalize-package-data "^2.3.4" + object-assign "^4.0.1" + read-pkg-up "^1.0.1" + redent "^1.0.0" + trim-newlines "^1.0.0" + +merge-descriptors@1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/merge-descriptors/-/merge-descriptors-1.0.1.tgz#b00aaa556dd8b44568150ec9d1b953f3f90cbb61" + +merge@^1.1.3: + version "1.2.0" + resolved "https://registry.yarnpkg.com/merge/-/merge-1.2.0.tgz#7531e39d4949c281a66b8c5a6e0265e8b05894da" + +methods@~1.1.2: + version "1.1.2" + resolved "https://registry.yarnpkg.com/methods/-/methods-1.1.2.tgz#5529a4d67654134edcc5266656835b0f851afcee" + +micromatch@^2.1.5, micromatch@^2.3.11: + version "2.3.11" + resolved "https://registry.yarnpkg.com/micromatch/-/micromatch-2.3.11.tgz#86677c97d1720b363431d04d0d15293bd38c1565" + dependencies: + arr-diff "^2.0.0" + array-unique "^0.2.1" + braces "^1.8.2" + expand-brackets "^0.1.4" + extglob "^0.3.1" + filename-regex "^2.0.0" + is-extglob "^1.0.0" + is-glob "^2.0.1" + kind-of "^3.0.2" + normalize-path "^2.0.1" + object.omit "^2.0.0" + parse-glob "^3.0.4" + regex-cache "^0.4.2" + +miller-rabin@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/miller-rabin/-/miller-rabin-4.0.0.tgz#4a62fb1d42933c05583982f4c716f6fb9e6c6d3d" + dependencies: + bn.js "^4.0.0" + brorand "^1.0.1" + +"mime-db@>= 1.29.0 < 2", mime-db@~1.30.0: + version "1.30.0" + resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.30.0.tgz#74c643da2dd9d6a45399963465b26d5ca7d71f01" + +mime-types@^2.1.12, mime-types@~2.1.15, mime-types@~2.1.16, mime-types@~2.1.7: + version "2.1.17" + resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.17.tgz#09d7a393f03e995a79f8af857b70a9e0ab16557a" + dependencies: + mime-db "~1.30.0" + +mime@1.3.4: + version "1.3.4" + resolved "https://registry.yarnpkg.com/mime/-/mime-1.3.4.tgz#115f9e3b6b3daf2959983cb38f149a2d40eb5d53" + +mime@1.3.x: + version "1.3.6" + resolved "https://registry.yarnpkg.com/mime/-/mime-1.3.6.tgz#591d84d3653a6b0b4a3b9df8de5aa8108e72e5e0" + +mime@^1.2.11, mime@^1.3.4: + version "1.4.0" + resolved "https://registry.yarnpkg.com/mime/-/mime-1.4.0.tgz#69e9e0db51d44f2a3b56e48b7817d7d137f1a343" + +mimic-fn@^1.0.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-1.1.0.tgz#e667783d92e89dbd342818b5230b9d62a672ad18" + +minimalistic-assert@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/minimalistic-assert/-/minimalistic-assert-1.0.0.tgz#702be2dda6b37f4836bcb3f5db56641b64a1d3d3" + +minimalistic-crypto-utils@^1.0.0, minimalistic-crypto-utils@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz#f6c00c1c0b082246e5c4d99dfb8c7c083b2b582a" + +minimatch@^3.0.0, minimatch@^3.0.2, minimatch@^3.0.3, minimatch@^3.0.4: + version "3.0.4" + resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.0.4.tgz#5166e286457f03306064be5497e8dbb0c3d32083" + dependencies: + brace-expansion "^1.1.7" + +minimist@0.0.8: + version "0.0.8" + resolved "https://registry.yarnpkg.com/minimist/-/minimist-0.0.8.tgz#857fcabfc3397d2625b8228262e86aa7a011b05d" + +minimist@^1.1.1, minimist@^1.1.3, minimist@^1.2.0, minimist@~1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.0.tgz#a35008b20f41383eec1fb914f4cd5df79a264284" + +minimist@~0.0.1: + version "0.0.10" + resolved "https://registry.yarnpkg.com/minimist/-/minimist-0.0.10.tgz#de3f98543dbf96082be48ad1a0c7cda836301dcf" + integrity sha1-3j+YVD2/lggr5IrRoMfNqDYwHc8= + +mkdirp@0.5.x, "mkdirp@>=0.5 0", mkdirp@^0.5.0, mkdirp@^0.5.1, mkdirp@~0.5.0, mkdirp@~0.5.1: + version "0.5.1" + resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-0.5.1.tgz#30057438eac6cf7f8c4767f38648d6697d75c903" + dependencies: + minimist "0.0.8" + +ms@2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/ms/-/ms-2.0.0.tgz#5608aeadfc00be6c2901df5f9861788de0d597c8" + +multicast-dns-service-types@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/multicast-dns-service-types/-/multicast-dns-service-types-1.1.0.tgz#899f11d9686e5e05cb91b35d5f0e63b773cfc901" + +multicast-dns@^6.0.1: + version "6.1.1" + resolved "https://registry.yarnpkg.com/multicast-dns/-/multicast-dns-6.1.1.tgz#6e7de86a570872ab17058adea7160bbeca814dde" + dependencies: + dns-packet "^1.0.1" + thunky "^0.1.0" + +mute-stream@0.0.7: + version "0.0.7" + resolved "https://registry.yarnpkg.com/mute-stream/-/mute-stream-0.0.7.tgz#3075ce93bc21b8fab43e1bc4da7e8115ed1e7bab" + +nan@^2.3.0: + version "2.7.0" + resolved "https://registry.yarnpkg.com/nan/-/nan-2.7.0.tgz#d95bf721ec877e08db276ed3fc6eb78f9083ad46" + +native-promise-only@^0.8.1: + version "0.8.1" + resolved "https://registry.yarnpkg.com/native-promise-only/-/native-promise-only-0.8.1.tgz#20a318c30cb45f71fe7adfbf7b21c99c1472ef11" + +natural-compare@^1.4.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/natural-compare/-/natural-compare-1.4.0.tgz#4abebfeed7541f2c27acfb29bdbbd15c8d5ba4f7" + +ncname@1.0.x: + version "1.0.0" + resolved "https://registry.yarnpkg.com/ncname/-/ncname-1.0.0.tgz#5b57ad18b1ca092864ef62b0b1ed8194f383b71c" + dependencies: + xml-char-classes "^1.0.0" + +negotiator@0.6.1: + version "0.6.1" + resolved "https://registry.yarnpkg.com/negotiator/-/negotiator-0.6.1.tgz#2b327184e8992101177b28563fb5e7102acd0ca9" + +neo-async@^2.6.0: + version "2.6.1" + resolved "https://registry.yarnpkg.com/neo-async/-/neo-async-2.6.1.tgz#ac27ada66167fa8849a6addd837f6b189ad2081c" + integrity sha512-iyam8fBuCUpWeKPGpaNMetEocMt364qkCsfL9JuhjXX6dRnguRVOfk2GZaDpPjcOKiiXCPINZC1GczQ7iTq3Zw== + +nise@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/nise/-/nise-1.0.1.tgz#0da92b10a854e97c0f496f6c2845a301280b3eef" + dependencies: + formatio "^1.2.0" + just-extend "^1.1.22" + lolex "^1.6.0" + path-to-regexp "^1.7.0" + +no-case@^2.2.0: + version "2.3.2" + resolved "https://registry.yarnpkg.com/no-case/-/no-case-2.3.2.tgz#60b813396be39b3f1288a4c1ed5d1e7d28b464ac" + dependencies: + lower-case "^1.1.1" + +node-fetch@^1.0.1: + version "1.7.3" + resolved "https://registry.yarnpkg.com/node-fetch/-/node-fetch-1.7.3.tgz#980f6f72d85211a5347c6b2bc18c5b84c3eb47ef" + dependencies: + encoding "^0.1.11" + is-stream "^1.0.1" + +node-forge@0.6.33: + version "0.6.33" + resolved "https://registry.yarnpkg.com/node-forge/-/node-forge-0.6.33.tgz#463811879f573d45155ad6a9f43dc296e8e85ebc" + +node-int64@^0.4.0: + version "0.4.0" + resolved "https://registry.yarnpkg.com/node-int64/-/node-int64-0.4.0.tgz#87a9065cdb355d3182d8f94ce11188b825c68a3b" + +node-libs-browser@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/node-libs-browser/-/node-libs-browser-2.0.0.tgz#a3a59ec97024985b46e958379646f96c4b616646" + dependencies: + assert "^1.1.1" + browserify-zlib "^0.1.4" + buffer "^4.3.0" + console-browserify "^1.1.0" + constants-browserify "^1.0.0" + crypto-browserify "^3.11.0" + domain-browser "^1.1.1" + events "^1.0.0" + https-browserify "0.0.1" + os-browserify "^0.2.0" + path-browserify "0.0.0" + process "^0.11.0" + punycode "^1.2.4" + querystring-es3 "^0.2.0" + readable-stream "^2.0.5" + stream-browserify "^2.0.1" + stream-http "^2.3.1" + string_decoder "^0.10.25" + timers-browserify "^2.0.2" + tty-browserify "0.0.0" + url "^0.11.0" + util "^0.10.3" + vm-browserify "0.0.4" + +node-notifier@^5.0.2: + version "5.1.2" + resolved "https://registry.yarnpkg.com/node-notifier/-/node-notifier-5.1.2.tgz#2fa9e12605fa10009d44549d6fcd8a63dde0e4ff" + dependencies: + growly "^1.3.0" + semver "^5.3.0" + shellwords "^0.1.0" + which "^1.2.12" + +node-pre-gyp@^0.6.36: + version "0.6.37" + resolved "https://registry.yarnpkg.com/node-pre-gyp/-/node-pre-gyp-0.6.37.tgz#3c872b236b2e266e4140578fe1ee88f693323a05" + dependencies: + mkdirp "^0.5.1" + nopt "^4.0.1" + npmlog "^4.0.2" + rc "^1.1.7" + request "^2.81.0" + rimraf "^2.6.1" + semver "^5.3.0" + tape "^4.6.3" + tar "^2.2.1" + tar-pack "^3.4.0" + +nopt@^4.0.1: + version "4.0.1" + resolved "https://registry.yarnpkg.com/nopt/-/nopt-4.0.1.tgz#d0d4685afd5415193c8c7505602d0d17cd64474d" + dependencies: + abbrev "1" + osenv "^0.1.4" + +normalize-package-data@^2.3.2, normalize-package-data@^2.3.4: + version "2.4.0" + resolved "https://registry.yarnpkg.com/normalize-package-data/-/normalize-package-data-2.4.0.tgz#12f95a307d58352075a04907b84ac8be98ac012f" + dependencies: + hosted-git-info "^2.1.4" + is-builtin-module "^1.0.0" + semver "2 || 3 || 4 || 5" + validate-npm-package-license "^3.0.1" + +normalize-path@^2.0.0, normalize-path@^2.0.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/normalize-path/-/normalize-path-2.1.1.tgz#1ab28b556e198363a8c1a6f7e6fa20137fe6aed9" + dependencies: + remove-trailing-separator "^1.0.1" + +normalize-range@^0.1.2: + version "0.1.2" + resolved "https://registry.yarnpkg.com/normalize-range/-/normalize-range-0.1.2.tgz#2d10c06bdfd312ea9777695a4d28439456b75942" + +normalize-url@^1.4.0: + version "1.9.1" + resolved "https://registry.yarnpkg.com/normalize-url/-/normalize-url-1.9.1.tgz#2cc0d66b31ea23036458436e3620d85954c66c3c" + dependencies: + object-assign "^4.0.1" + prepend-http "^1.0.0" + query-string "^4.1.0" + sort-keys "^1.0.0" + +npm-run-path@^2.0.0: + version "2.0.2" + resolved "https://registry.yarnpkg.com/npm-run-path/-/npm-run-path-2.0.2.tgz#35a9232dfa35d7067b4cb2ddf2357b1871536c5f" + dependencies: + path-key "^2.0.0" + +npmlog@^4.0.2: + version "4.1.2" + resolved "https://registry.yarnpkg.com/npmlog/-/npmlog-4.1.2.tgz#08a7f2a8bf734604779a9efa4ad5cc717abb954b" + dependencies: + are-we-there-yet "~1.1.2" + console-control-strings "~1.1.0" + gauge "~2.7.3" + set-blocking "~2.0.0" + +nth-check@~1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/nth-check/-/nth-check-1.0.1.tgz#9929acdf628fc2c41098deab82ac580cf149aae4" + dependencies: + boolbase "~1.0.0" + +num2fraction@^1.2.2: + version "1.2.2" + resolved "https://registry.yarnpkg.com/num2fraction/-/num2fraction-1.2.2.tgz#6f682b6a027a4e9ddfa4564cd2589d1d4e669ede" + +number-is-nan@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/number-is-nan/-/number-is-nan-1.0.1.tgz#097b602b53422a522c1afb8790318336941a011d" + +"nwmatcher@>= 1.3.9 < 2.0.0": + version "1.4.1" + resolved "https://registry.yarnpkg.com/nwmatcher/-/nwmatcher-1.4.1.tgz#7ae9b07b0ea804db7e25f05cb5fe4097d4e4949f" + +oauth-sign@~0.8.1: + version "0.8.2" + resolved "https://registry.yarnpkg.com/oauth-sign/-/oauth-sign-0.8.2.tgz#46a6ab7f0aead8deae9ec0565780b7d4efeb9d43" + +object-assign@^4.0.1, object-assign@^4.1.0: + version "4.1.1" + resolved "https://registry.yarnpkg.com/object-assign/-/object-assign-4.1.1.tgz#2109adc7965887cfc05cbbd442cac8bfbb360863" + +object-inspect@~1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/object-inspect/-/object-inspect-1.3.0.tgz#5b1eb8e6742e2ee83342a637034d844928ba2f6d" + +object-is@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/object-is/-/object-is-1.0.1.tgz#0aa60ec9989a0b3ed795cf4d06f62cf1ad6539b6" + +object-keys@^1.0.10, object-keys@^1.0.8: + version "1.0.11" + resolved "https://registry.yarnpkg.com/object-keys/-/object-keys-1.0.11.tgz#c54601778ad560f1142ce0e01bcca8b56d13426d" + +object.assign@^4.0.4: + version "4.0.4" + resolved "https://registry.yarnpkg.com/object.assign/-/object.assign-4.0.4.tgz#b1c9cc044ef1b9fe63606fc141abbb32e14730cc" + dependencies: + define-properties "^1.1.2" + function-bind "^1.1.0" + object-keys "^1.0.10" + +object.entries@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/object.entries/-/object.entries-1.0.4.tgz#1bf9a4dd2288f5b33f3a993d257661f05d161a5f" + dependencies: + define-properties "^1.1.2" + es-abstract "^1.6.1" + function-bind "^1.1.0" + has "^1.0.1" + +object.omit@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/object.omit/-/object.omit-2.0.1.tgz#1a9c744829f39dbb858c76ca3579ae2a54ebd1fa" + dependencies: + for-own "^0.1.4" + is-extendable "^0.1.1" + +object.values@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/object.values/-/object.values-1.0.4.tgz#e524da09b4f66ff05df457546ec72ac99f13069a" + dependencies: + define-properties "^1.1.2" + es-abstract "^1.6.1" + function-bind "^1.1.0" + has "^1.0.1" + +obuf@^1.0.0, obuf@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/obuf/-/obuf-1.1.1.tgz#104124b6c602c6796881a042541d36db43a5264e" + +on-finished@~2.3.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/on-finished/-/on-finished-2.3.0.tgz#20f1336481b083cd75337992a16971aa2d906947" + dependencies: + ee-first "1.1.1" + +on-headers@~1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/on-headers/-/on-headers-1.0.1.tgz#928f5d0f470d49342651ea6794b0857c100693f7" + +once@^1.3.0, once@^1.3.3, once@^1.4.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1" + dependencies: + wrappy "1" + +onetime@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/onetime/-/onetime-2.0.1.tgz#067428230fd67443b2794b22bba528b6867962d4" + dependencies: + mimic-fn "^1.0.0" + +opn@4.0.2: + version "4.0.2" + resolved "https://registry.yarnpkg.com/opn/-/opn-4.0.2.tgz#7abc22e644dff63b0a96d5ab7f2790c0f01abc95" + dependencies: + object-assign "^4.0.1" + pinkie-promise "^2.0.0" + +optimist@^0.6.1: + version "0.6.1" + resolved "https://registry.yarnpkg.com/optimist/-/optimist-0.6.1.tgz#da3ea74686fa21a19a111c326e90eb15a0196686" + integrity sha1-2j6nRob6IaGaERwybpDrFaAZZoY= + dependencies: + minimist "~0.0.1" + wordwrap "~0.0.2" + +optionator@^0.8.1, optionator@^0.8.2: + version "0.8.2" + resolved "https://registry.yarnpkg.com/optionator/-/optionator-0.8.2.tgz#364c5e409d3f4d6301d6c0b4c05bba50180aeb64" + dependencies: + deep-is "~0.1.3" + fast-levenshtein "~2.0.4" + levn "~0.3.0" + prelude-ls "~1.1.2" + type-check "~0.3.2" + wordwrap "~1.0.0" + +original@>=0.0.5: + version "1.0.0" + resolved "https://registry.yarnpkg.com/original/-/original-1.0.0.tgz#9147f93fa1696d04be61e01bd50baeaca656bd3b" + dependencies: + url-parse "1.0.x" + +os-browserify@^0.2.0: + version "0.2.1" + resolved "https://registry.yarnpkg.com/os-browserify/-/os-browserify-0.2.1.tgz#63fc4ccee5d2d7763d26bbf8601078e6c2e0044f" + +os-homedir@^1.0.0: + version "1.0.2" + resolved "https://registry.yarnpkg.com/os-homedir/-/os-homedir-1.0.2.tgz#ffbc4988336e0e833de0c168c7ef152121aa7fb3" + +os-locale@^1.4.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/os-locale/-/os-locale-1.4.0.tgz#20f9f17ae29ed345e8bde583b13d2009803c14d9" + dependencies: + lcid "^1.0.0" + +os-locale@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/os-locale/-/os-locale-2.1.0.tgz#42bc2900a6b5b8bd17376c8e882b65afccf24bf2" + dependencies: + execa "^0.7.0" + lcid "^1.0.0" + mem "^1.1.0" + +os-tmpdir@^1.0.0, os-tmpdir@^1.0.1, os-tmpdir@~1.0.1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/os-tmpdir/-/os-tmpdir-1.0.2.tgz#bbe67406c79aa85c5cfec766fe5734555dfa1274" + +osenv@^0.1.4: + version "0.1.4" + resolved "https://registry.yarnpkg.com/osenv/-/osenv-0.1.4.tgz#42fe6d5953df06c8064be6f176c3d05aaaa34644" + dependencies: + os-homedir "^1.0.0" + os-tmpdir "^1.0.0" + +p-cancelable@^0.3.0: + version "0.3.0" + resolved "https://registry.yarnpkg.com/p-cancelable/-/p-cancelable-0.3.0.tgz#b9e123800bcebb7ac13a479be195b507b98d30fa" + +p-finally@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/p-finally/-/p-finally-1.0.0.tgz#3fbcfb15b899a44123b34b6dcc18b724336a2cae" + +p-limit@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-1.1.0.tgz#b07ff2d9a5d88bec806035895a2bab66a27988bc" + +p-locate@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-2.0.0.tgz#20a0103b222a70c8fd39cc2e580680f3dde5ec43" + dependencies: + p-limit "^1.1.0" + +p-map@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/p-map/-/p-map-1.1.1.tgz#05f5e4ae97a068371bc2a5cc86bfbdbc19c4ae7a" + +pako@~0.2.0: + version "0.2.9" + resolved "https://registry.yarnpkg.com/pako/-/pako-0.2.9.tgz#f3f7522f4ef782348da8161bad9ecfd51bf83a75" + +param-case@2.1.x: + version "2.1.1" + resolved "https://registry.yarnpkg.com/param-case/-/param-case-2.1.1.tgz#df94fd8cf6531ecf75e6bef9a0858fbc72be2247" + dependencies: + no-case "^2.2.0" + +parse-asn1@^5.0.0: + version "5.1.0" + resolved "https://registry.yarnpkg.com/parse-asn1/-/parse-asn1-5.1.0.tgz#37c4f9b7ed3ab65c74817b5f2480937fbf97c712" + dependencies: + asn1.js "^4.0.0" + browserify-aes "^1.0.0" + create-hash "^1.1.0" + evp_bytestokey "^1.0.0" + pbkdf2 "^3.0.3" + +parse-glob@^3.0.4: + version "3.0.4" + resolved "https://registry.yarnpkg.com/parse-glob/-/parse-glob-3.0.4.tgz#b2c376cfb11f35513badd173ef0bb6e3a388391c" + dependencies: + glob-base "^0.3.0" + is-dotfile "^1.0.0" + is-extglob "^1.0.0" + is-glob "^2.0.0" + +parse-json@^2.2.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/parse-json/-/parse-json-2.2.0.tgz#f480f40434ef80741f8469099f8dea18f55a4dc9" + dependencies: + error-ex "^1.2.0" + +parse5@^1.5.1: + version "1.5.1" + resolved "https://registry.yarnpkg.com/parse5/-/parse5-1.5.1.tgz#9b7f3b0de32be78dc2401b17573ccaf0f6f59d94" + +parseurl@~1.3.1: + version "1.3.2" + resolved "https://registry.yarnpkg.com/parseurl/-/parseurl-1.3.2.tgz#fc289d4ed8993119460c156253262cdc8de65bf3" + +path-browserify@0.0.0: + version "0.0.0" + resolved "https://registry.yarnpkg.com/path-browserify/-/path-browserify-0.0.0.tgz#a0b870729aae214005b7d5032ec2cbbb0fb4451a" + +path-exists@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-2.1.0.tgz#0feb6c64f0fc518d9a754dd5efb62c7022761f4b" + dependencies: + pinkie-promise "^2.0.0" + +path-exists@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-3.0.0.tgz#ce0ebeaa5f78cb18925ea7d810d7b59b010fd515" + +path-is-absolute@^1.0.0, path-is-absolute@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/path-is-absolute/-/path-is-absolute-1.0.1.tgz#174b9268735534ffbc7ace6bf53a5a9e1b5c5f5f" + +path-is-inside@^1.0.1, path-is-inside@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/path-is-inside/-/path-is-inside-1.0.2.tgz#365417dede44430d1c11af61027facf074bdfc53" + +path-key@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/path-key/-/path-key-2.0.1.tgz#411cadb574c5a140d3a4b1910d40d80cc9f40b40" + +path-parse@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/path-parse/-/path-parse-1.0.5.tgz#3c1adf871ea9cd6c9431b6ea2bd74a0ff055c4c1" + +path-to-regexp@0.1.7: + version "0.1.7" + resolved "https://registry.yarnpkg.com/path-to-regexp/-/path-to-regexp-0.1.7.tgz#df604178005f522f15eb4490e7247a1bfaa67f8c" + +path-to-regexp@^1.7.0: + version "1.7.0" + resolved "https://registry.yarnpkg.com/path-to-regexp/-/path-to-regexp-1.7.0.tgz#59fde0f435badacba103a84e9d3bc64e96b9937d" + dependencies: + isarray "0.0.1" + +path-type@^1.0.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/path-type/-/path-type-1.1.0.tgz#59c44f7ee491da704da415da5a4070ba4f8fe441" + dependencies: + graceful-fs "^4.1.2" + pify "^2.0.0" + pinkie-promise "^2.0.0" + +path-type@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/path-type/-/path-type-2.0.0.tgz#f012ccb8415b7096fc2daa1054c3d72389594c73" + dependencies: + pify "^2.0.0" + +pbkdf2@^3.0.3: + version "3.0.14" + resolved "https://registry.yarnpkg.com/pbkdf2/-/pbkdf2-3.0.14.tgz#a35e13c64799b06ce15320f459c230e68e73bade" + dependencies: + create-hash "^1.1.2" + create-hmac "^1.1.4" + ripemd160 "^2.0.1" + safe-buffer "^5.0.1" + sha.js "^2.4.8" + +performance-now@^0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/performance-now/-/performance-now-0.2.0.tgz#33ef30c5c77d4ea21c5a53869d91b56d8f2555e5" + +pify@^2.0.0, pify@^2.3.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/pify/-/pify-2.3.0.tgz#ed141a6ac043a849ea588498e7dca8b15330e90c" + +pify@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/pify/-/pify-3.0.0.tgz#e5a4acd2c101fdf3d9a4d07f0dbc4db49dd28176" + +pinkie-promise@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/pinkie-promise/-/pinkie-promise-2.0.1.tgz#2135d6dfa7a358c069ac9b178776288228450ffa" + dependencies: + pinkie "^2.0.0" + +pinkie@^2.0.0: + version "2.0.4" + resolved "https://registry.yarnpkg.com/pinkie/-/pinkie-2.0.4.tgz#72556b80cfa0d48a974e80e77248e80ed4f7f870" + +pkg-dir@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/pkg-dir/-/pkg-dir-1.0.0.tgz#7a4b508a8d5bb2d629d447056ff4e9c9314cf3d4" + dependencies: + find-up "^1.0.0" + +pkg-dir@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/pkg-dir/-/pkg-dir-2.0.0.tgz#f6d5d1109e19d63edf428e0bd57e12777615334b" + dependencies: + find-up "^2.1.0" + +pluralize@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/pluralize/-/pluralize-4.0.0.tgz#59b708c1c0190a2f692f1c7618c446b052fd1762" + +portfinder@^1.0.9: + version "1.0.13" + resolved "https://registry.yarnpkg.com/portfinder/-/portfinder-1.0.13.tgz#bb32ecd87c27104ae6ee44b5a3ccbf0ebb1aede9" + dependencies: + async "^1.5.2" + debug "^2.2.0" + mkdirp "0.5.x" + +postcss-calc@^5.2.0: + version "5.3.1" + resolved "https://registry.yarnpkg.com/postcss-calc/-/postcss-calc-5.3.1.tgz#77bae7ca928ad85716e2fda42f261bf7c1d65b5e" + dependencies: + postcss "^5.0.2" + postcss-message-helpers "^2.0.0" + reduce-css-calc "^1.2.6" + +postcss-colormin@^2.1.8: + version "2.2.2" + resolved "https://registry.yarnpkg.com/postcss-colormin/-/postcss-colormin-2.2.2.tgz#6631417d5f0e909a3d7ec26b24c8a8d1e4f96e4b" + dependencies: + colormin "^1.0.5" + postcss "^5.0.13" + postcss-value-parser "^3.2.3" + +postcss-convert-values@^2.3.4: + version "2.6.1" + resolved "https://registry.yarnpkg.com/postcss-convert-values/-/postcss-convert-values-2.6.1.tgz#bbd8593c5c1fd2e3d1c322bb925dcae8dae4d62d" + dependencies: + postcss "^5.0.11" + postcss-value-parser "^3.1.2" + +postcss-discard-comments@^2.0.4: + version "2.0.4" + resolved "https://registry.yarnpkg.com/postcss-discard-comments/-/postcss-discard-comments-2.0.4.tgz#befe89fafd5b3dace5ccce51b76b81514be00e3d" + dependencies: + postcss "^5.0.14" + +postcss-discard-duplicates@^2.0.1: + version "2.1.0" + resolved "https://registry.yarnpkg.com/postcss-discard-duplicates/-/postcss-discard-duplicates-2.1.0.tgz#b9abf27b88ac188158a5eb12abcae20263b91932" + dependencies: + postcss "^5.0.4" + +postcss-discard-empty@^2.0.1: + version "2.1.0" + resolved "https://registry.yarnpkg.com/postcss-discard-empty/-/postcss-discard-empty-2.1.0.tgz#d2b4bd9d5ced5ebd8dcade7640c7d7cd7f4f92b5" + dependencies: + postcss "^5.0.14" + +postcss-discard-overridden@^0.1.1: + version "0.1.1" + resolved "https://registry.yarnpkg.com/postcss-discard-overridden/-/postcss-discard-overridden-0.1.1.tgz#8b1eaf554f686fb288cd874c55667b0aa3668d58" + dependencies: + postcss "^5.0.16" + +postcss-discard-unused@^2.2.1: + version "2.2.3" + resolved "https://registry.yarnpkg.com/postcss-discard-unused/-/postcss-discard-unused-2.2.3.tgz#bce30b2cc591ffc634322b5fb3464b6d934f4433" + dependencies: + postcss "^5.0.14" + uniqs "^2.0.0" + +postcss-filter-plugins@^2.0.0: + version "2.0.2" + resolved "https://registry.yarnpkg.com/postcss-filter-plugins/-/postcss-filter-plugins-2.0.2.tgz#6d85862534d735ac420e4a85806e1f5d4286d84c" + dependencies: + postcss "^5.0.4" + uniqid "^4.0.0" + +postcss-less@^1.0.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/postcss-less/-/postcss-less-1.1.0.tgz#bdcc76be64c4324d873fbc5cd9fa2e799e4305fa" + dependencies: + postcss "^5.2.16" + +postcss-merge-idents@^2.1.5: + version "2.1.7" + resolved "https://registry.yarnpkg.com/postcss-merge-idents/-/postcss-merge-idents-2.1.7.tgz#4c5530313c08e1d5b3bbf3d2bbc747e278eea270" + dependencies: + has "^1.0.1" + postcss "^5.0.10" + postcss-value-parser "^3.1.1" + +postcss-merge-longhand@^2.0.1: + version "2.0.2" + resolved "https://registry.yarnpkg.com/postcss-merge-longhand/-/postcss-merge-longhand-2.0.2.tgz#23d90cd127b0a77994915332739034a1a4f3d658" + dependencies: + postcss "^5.0.4" + +postcss-merge-rules@^2.0.3: + version "2.1.2" + resolved "https://registry.yarnpkg.com/postcss-merge-rules/-/postcss-merge-rules-2.1.2.tgz#d1df5dfaa7b1acc3be553f0e9e10e87c61b5f721" + dependencies: + browserslist "^1.5.2" + caniuse-api "^1.5.2" + postcss "^5.0.4" + postcss-selector-parser "^2.2.2" + vendors "^1.0.0" + +postcss-message-helpers@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/postcss-message-helpers/-/postcss-message-helpers-2.0.0.tgz#a4f2f4fab6e4fe002f0aed000478cdf52f9ba60e" + +postcss-minify-font-values@^1.0.2: + version "1.0.5" + resolved "https://registry.yarnpkg.com/postcss-minify-font-values/-/postcss-minify-font-values-1.0.5.tgz#4b58edb56641eba7c8474ab3526cafd7bbdecb69" + dependencies: + object-assign "^4.0.1" + postcss "^5.0.4" + postcss-value-parser "^3.0.2" + +postcss-minify-gradients@^1.0.1: + version "1.0.5" + resolved "https://registry.yarnpkg.com/postcss-minify-gradients/-/postcss-minify-gradients-1.0.5.tgz#5dbda11373703f83cfb4a3ea3881d8d75ff5e6e1" + dependencies: + postcss "^5.0.12" + postcss-value-parser "^3.3.0" + +postcss-minify-params@^1.0.4: + version "1.2.2" + resolved "https://registry.yarnpkg.com/postcss-minify-params/-/postcss-minify-params-1.2.2.tgz#ad2ce071373b943b3d930a3fa59a358c28d6f1f3" + dependencies: + alphanum-sort "^1.0.1" + postcss "^5.0.2" + postcss-value-parser "^3.0.2" + uniqs "^2.0.0" + +postcss-minify-selectors@^2.0.4: + version "2.1.1" + resolved "https://registry.yarnpkg.com/postcss-minify-selectors/-/postcss-minify-selectors-2.1.1.tgz#b2c6a98c0072cf91b932d1a496508114311735bf" + dependencies: + alphanum-sort "^1.0.2" + has "^1.0.1" + postcss "^5.0.14" + postcss-selector-parser "^2.0.0" + +postcss-modules-extract-imports@^1.0.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/postcss-modules-extract-imports/-/postcss-modules-extract-imports-1.2.0.tgz#66140ecece38ef06bf0d3e355d69bf59d141ea85" + dependencies: + postcss "^6.0.1" + +postcss-modules-local-by-default@^1.0.1: + version "1.2.0" + resolved "https://registry.yarnpkg.com/postcss-modules-local-by-default/-/postcss-modules-local-by-default-1.2.0.tgz#f7d80c398c5a393fa7964466bd19500a7d61c069" + dependencies: + css-selector-tokenizer "^0.7.0" + postcss "^6.0.1" + +postcss-modules-scope@^1.0.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/postcss-modules-scope/-/postcss-modules-scope-1.1.0.tgz#d6ea64994c79f97b62a72b426fbe6056a194bb90" + dependencies: + css-selector-tokenizer "^0.7.0" + postcss "^6.0.1" + +postcss-modules-values@^1.1.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/postcss-modules-values/-/postcss-modules-values-1.3.0.tgz#ecffa9d7e192518389f42ad0e83f72aec456ea20" + dependencies: + icss-replace-symbols "^1.1.0" + postcss "^6.0.1" + +postcss-normalize-charset@^1.1.0: + version "1.1.1" + resolved "https://registry.yarnpkg.com/postcss-normalize-charset/-/postcss-normalize-charset-1.1.1.tgz#ef9ee71212d7fe759c78ed162f61ed62b5cb93f1" + dependencies: + postcss "^5.0.5" + +postcss-normalize-url@^3.0.7: + version "3.0.8" + resolved "https://registry.yarnpkg.com/postcss-normalize-url/-/postcss-normalize-url-3.0.8.tgz#108f74b3f2fcdaf891a2ffa3ea4592279fc78222" + dependencies: + is-absolute-url "^2.0.0" + normalize-url "^1.4.0" + postcss "^5.0.14" + postcss-value-parser "^3.2.3" + +postcss-ordered-values@^2.1.0: + version "2.2.3" + resolved "https://registry.yarnpkg.com/postcss-ordered-values/-/postcss-ordered-values-2.2.3.tgz#eec6c2a67b6c412a8db2042e77fe8da43f95c11d" + dependencies: + postcss "^5.0.4" + postcss-value-parser "^3.0.1" + +postcss-reduce-idents@^2.2.2: + version "2.4.0" + resolved "https://registry.yarnpkg.com/postcss-reduce-idents/-/postcss-reduce-idents-2.4.0.tgz#c2c6d20cc958284f6abfbe63f7609bf409059ad3" + dependencies: + postcss "^5.0.4" + postcss-value-parser "^3.0.2" + +postcss-reduce-initial@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/postcss-reduce-initial/-/postcss-reduce-initial-1.0.1.tgz#68f80695f045d08263a879ad240df8dd64f644ea" + dependencies: + postcss "^5.0.4" + +postcss-reduce-transforms@^1.0.3: + version "1.0.4" + resolved "https://registry.yarnpkg.com/postcss-reduce-transforms/-/postcss-reduce-transforms-1.0.4.tgz#ff76f4d8212437b31c298a42d2e1444025771ae1" + dependencies: + has "^1.0.1" + postcss "^5.0.8" + postcss-value-parser "^3.0.1" + +postcss-selector-parser@^2.0.0, postcss-selector-parser@^2.2.2: + version "2.2.3" + resolved "https://registry.yarnpkg.com/postcss-selector-parser/-/postcss-selector-parser-2.2.3.tgz#f9437788606c3c9acee16ffe8d8b16297f27bb90" + dependencies: + flatten "^1.0.2" + indexes-of "^1.0.1" + uniq "^1.0.1" + +postcss-svgo@^2.1.1: + version "2.1.6" + resolved "https://registry.yarnpkg.com/postcss-svgo/-/postcss-svgo-2.1.6.tgz#b6df18aa613b666e133f08adb5219c2684ac108d" + dependencies: + is-svg "^2.0.0" + postcss "^5.0.14" + postcss-value-parser "^3.2.3" + svgo "^0.7.0" + +postcss-unique-selectors@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/postcss-unique-selectors/-/postcss-unique-selectors-2.0.2.tgz#981d57d29ddcb33e7b1dfe1fd43b8649f933ca1d" + dependencies: + alphanum-sort "^1.0.1" + postcss "^5.0.4" + uniqs "^2.0.0" + +postcss-value-parser@^3.0.1, postcss-value-parser@^3.0.2, postcss-value-parser@^3.1.1, postcss-value-parser@^3.1.2, postcss-value-parser@^3.2.3, postcss-value-parser@^3.3.0: + version "3.3.0" + resolved "https://registry.yarnpkg.com/postcss-value-parser/-/postcss-value-parser-3.3.0.tgz#87f38f9f18f774a4ab4c8a232f5c5ce8872a9d15" + +postcss-values-parser@~1.2.2: + version "1.2.2" + resolved "https://registry.yarnpkg.com/postcss-values-parser/-/postcss-values-parser-1.2.2.tgz#f5eebdd7f6fe310e2e448b9cc03bfed8726ca271" + dependencies: + flatten "^1.0.2" + indexes-of "^1.0.1" + uniq "^1.0.1" + +postcss-zindex@^2.0.1: + version "2.2.0" + resolved "https://registry.yarnpkg.com/postcss-zindex/-/postcss-zindex-2.2.0.tgz#d2109ddc055b91af67fc4cb3b025946639d2af22" + dependencies: + has "^1.0.1" + postcss "^5.0.4" + uniqs "^2.0.0" + +postcss@^5.0.10, postcss@^5.0.11, postcss@^5.0.12, postcss@^5.0.13, postcss@^5.0.14, postcss@^5.0.16, postcss@^5.0.2, postcss@^5.0.4, postcss@^5.0.5, postcss@^5.0.6, postcss@^5.0.8, postcss@^5.2.16: + version "5.2.17" + resolved "https://registry.yarnpkg.com/postcss/-/postcss-5.2.17.tgz#cf4f597b864d65c8a492b2eabe9d706c879c388b" + dependencies: + chalk "^1.1.3" + js-base64 "^2.1.9" + source-map "^0.5.6" + supports-color "^3.2.3" + +postcss@^6.0.0: + version "6.0.12" + resolved "https://registry.yarnpkg.com/postcss/-/postcss-6.0.12.tgz#6b0155089d2d212f7bd6a0cecd4c58c007403535" + dependencies: + chalk "^2.1.0" + source-map "^0.5.7" + supports-color "^4.4.0" + +postcss@^6.0.1: + version "6.0.11" + resolved "https://registry.yarnpkg.com/postcss/-/postcss-6.0.11.tgz#f48db210b1d37a7f7ab6499b7a54982997ab6f72" + dependencies: + chalk "^2.1.0" + source-map "^0.5.7" + supports-color "^4.4.0" + +preact-compat-enzyme@^0.2.5: + version "0.2.5" + resolved "https://registry.yarnpkg.com/preact-compat-enzyme/-/preact-compat-enzyme-0.2.5.tgz#1cad6c5c99f749d079fcdccf695470d88331d76f" + dependencies: + preact "8.x" + preact-compat "3.x" + +preact-compat@3.x, preact-compat@^3.11.0, preact-compat@^3.16.0: + version "3.17.0" + resolved "https://registry.yarnpkg.com/preact-compat/-/preact-compat-3.17.0.tgz#528cfdfc301190c1a0f47567336be1f4be0266b3" + dependencies: + immutability-helper "^2.1.2" + preact-render-to-string "^3.6.0" + preact-transition-group "^1.1.0" + prop-types "^15.5.8" + standalone-react-addons-pure-render-mixin "^0.1.1" + +preact-render-to-string@^3.6.0, preact-render-to-string@^3.6.3: + version "3.6.3" + resolved "https://registry.yarnpkg.com/preact-render-to-string/-/preact-render-to-string-3.6.3.tgz#481d0d5bdac9192d3347557437d5cd00aa312043" + dependencies: + pretty-format "^3.5.1" + +preact-test-utils@^0.1.3: + version "0.1.3" + resolved "https://registry.yarnpkg.com/preact-test-utils/-/preact-test-utils-0.1.3.tgz#26893e5a8ad7f4d8901bca02086b014f10b250e7" + dependencies: + preact-compat "^3.11.0" + simulate-event "^1.4.0" + +preact-transition-group@^1.1.0: + version "1.1.1" + resolved "https://registry.yarnpkg.com/preact-transition-group/-/preact-transition-group-1.1.1.tgz#f0a49327ea515ece34ea2be864c4a7d29e5d6e10" + +preact@8.x, preact@^8.2.1: + version "8.2.5" + resolved "https://registry.yarnpkg.com/preact/-/preact-8.2.5.tgz#cbfa3962a8012768159f6d01d46f9c1eb3213c0a" + +prelude-ls@~1.1.2: + version "1.1.2" + resolved "https://registry.yarnpkg.com/prelude-ls/-/prelude-ls-1.1.2.tgz#21932a549f5e52ffd9a827f570e04be62a97da54" + +prepend-http@^1.0.0: + version "1.0.4" + resolved "https://registry.yarnpkg.com/prepend-http/-/prepend-http-1.0.4.tgz#d4f4562b0ce3696e41ac52d0e002e57a635dc6dc" + +preserve@^0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/preserve/-/preserve-0.2.0.tgz#815ed1f6ebc65926f865b310c0713bcb3315ce4b" + +prettier@^1.6.0: + version "1.6.1" + resolved "https://registry.yarnpkg.com/prettier/-/prettier-1.6.1.tgz#850f411a3116226193e32ea5acfc21c0f9a76d7d" + +pretty-error@^2.0.2: + version "2.1.1" + resolved "https://registry.yarnpkg.com/pretty-error/-/pretty-error-2.1.1.tgz#5f4f87c8f91e5ae3f3ba87ab4cf5e03b1a17f1a3" + dependencies: + renderkid "^2.0.1" + utila "~0.4" + +pretty-format@^21.0.2: + version "21.0.2" + resolved "https://registry.yarnpkg.com/pretty-format/-/pretty-format-21.0.2.tgz#76adcebd836c41ccd2e6b626e70f63050d2a3534" + dependencies: + ansi-regex "^3.0.0" + ansi-styles "^3.2.0" + +pretty-format@^3.5.1: + version "3.8.0" + resolved "https://registry.yarnpkg.com/pretty-format/-/pretty-format-3.8.0.tgz#bfbed56d5e9a776645f4b1ff7aa1a3ac4fa3c385" + +private@^0.1.6, private@^0.1.7: + version "0.1.7" + resolved "https://registry.yarnpkg.com/private/-/private-0.1.7.tgz#68ce5e8a1ef0a23bb570cc28537b5332aba63ef1" + +process-nextick-args@~1.0.6: + version "1.0.7" + resolved "https://registry.yarnpkg.com/process-nextick-args/-/process-nextick-args-1.0.7.tgz#150e20b756590ad3f91093f25a4f2ad8bff30ba3" + +process@^0.11.0: + version "0.11.10" + resolved "https://registry.yarnpkg.com/process/-/process-0.11.10.tgz#7332300e840161bda3e69a1d1d91a7d4bc16f182" + +progress@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/progress/-/progress-2.0.0.tgz#8a1be366bf8fc23db2bd23f10c6fe920b4389d1f" + +promise@^7.1.1: + version "7.3.1" + resolved "https://registry.yarnpkg.com/promise/-/promise-7.3.1.tgz#064b72602b18f90f29192b8b1bc418ffd1ebd3bf" + dependencies: + asap "~2.0.3" + +prop-types@^15.5.10, prop-types@^15.5.4, prop-types@^15.5.6, prop-types@^15.5.8: + version "15.5.10" + resolved "https://registry.yarnpkg.com/prop-types/-/prop-types-15.5.10.tgz#2797dfc3126182e3a95e3dfbb2e893ddd7456154" + dependencies: + fbjs "^0.8.9" + loose-envify "^1.3.1" + +proxy-addr@~1.1.5: + version "1.1.5" + resolved "https://registry.yarnpkg.com/proxy-addr/-/proxy-addr-1.1.5.tgz#71c0ee3b102de3f202f3b64f608d173fcba1a918" + dependencies: + forwarded "~0.1.0" + ipaddr.js "1.4.0" + +prr@~0.0.0: + version "0.0.0" + resolved "https://registry.yarnpkg.com/prr/-/prr-0.0.0.tgz#1a84b85908325501411853d0081ee3fa86e2926a" + +pseudomap@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/pseudomap/-/pseudomap-1.0.2.tgz#f052a28da70e618917ef0a8ac34c1ae5a68286b3" + +public-encrypt@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/public-encrypt/-/public-encrypt-4.0.0.tgz#39f699f3a46560dd5ebacbca693caf7c65c18cc6" + dependencies: + bn.js "^4.1.0" + browserify-rsa "^4.0.0" + create-hash "^1.1.0" + parse-asn1 "^5.0.0" + randombytes "^2.0.1" + +punycode@1.3.2: + version "1.3.2" + resolved "https://registry.yarnpkg.com/punycode/-/punycode-1.3.2.tgz#9653a036fb7c1ee42342f2325cceefea3926c48d" + +punycode@^1.2.4, punycode@^1.4.1: + version "1.4.1" + resolved "https://registry.yarnpkg.com/punycode/-/punycode-1.4.1.tgz#c0d5a63b2718800ad8e1eb0fa5269c84dd41845e" + +q@^1.1.2: + version "1.5.0" + resolved "https://registry.yarnpkg.com/q/-/q-1.5.0.tgz#dd01bac9d06d30e6f219aecb8253ee9ebdc308f1" + +qs@6.5.0: + version "6.5.0" + resolved "https://registry.yarnpkg.com/qs/-/qs-6.5.0.tgz#8d04954d364def3efc55b5a0793e1e2c8b1e6e49" + +qs@~6.4.0: + version "6.4.0" + resolved "https://registry.yarnpkg.com/qs/-/qs-6.4.0.tgz#13e26d28ad6b0ffaa91312cd3bf708ed351e7233" + +query-string@^4.1.0: + version "4.3.4" + resolved "https://registry.yarnpkg.com/query-string/-/query-string-4.3.4.tgz#bbb693b9ca915c232515b228b1a02b609043dbeb" + dependencies: + object-assign "^4.1.0" + strict-uri-encode "^1.0.0" + +query-string@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/query-string/-/query-string-5.0.0.tgz#fbdf7004b4d2aff792f9871981b7a2794f555947" + dependencies: + decode-uri-component "^0.2.0" + object-assign "^4.1.0" + strict-uri-encode "^1.0.0" + +querystring-es3@^0.2.0: + version "0.2.1" + resolved "https://registry.yarnpkg.com/querystring-es3/-/querystring-es3-0.2.1.tgz#9ec61f79049875707d69414596fd907a4d711e73" + +querystring@0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/querystring/-/querystring-0.2.0.tgz#b209849203bb25df820da756e747005878521620" + +querystringify@0.0.x: + version "0.0.4" + resolved "https://registry.yarnpkg.com/querystringify/-/querystringify-0.0.4.tgz#0cf7f84f9463ff0ae51c4c4b142d95be37724d9c" + +querystringify@~1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/querystringify/-/querystringify-1.0.0.tgz#6286242112c5b712fa654e526652bf6a13ff05cb" + +randomatic@^1.1.3: + version "1.1.7" + resolved "https://registry.yarnpkg.com/randomatic/-/randomatic-1.1.7.tgz#c7abe9cc8b87c0baa876b19fde83fd464797e38c" + dependencies: + is-number "^3.0.0" + kind-of "^4.0.0" + +randombytes@^2.0.0, randombytes@^2.0.1: + version "2.0.5" + resolved "https://registry.yarnpkg.com/randombytes/-/randombytes-2.0.5.tgz#dc009a246b8d09a177b4b7a0ae77bc570f4b1b79" + dependencies: + safe-buffer "^5.1.0" + +range-parser@^1.0.3, range-parser@~1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/range-parser/-/range-parser-1.2.0.tgz#f49be6b487894ddc40dcc94a322f611092e00d5e" + +rc@^1.1.7: + version "1.2.1" + resolved "https://registry.yarnpkg.com/rc/-/rc-1.2.1.tgz#2e03e8e42ee450b8cb3dce65be1bf8974e1dfd95" + dependencies: + deep-extend "~0.4.0" + ini "~1.3.0" + minimist "^1.2.0" + strip-json-comments "~2.0.1" + +rcfinder@^0.1.8: + version "0.1.9" + resolved "https://registry.yarnpkg.com/rcfinder/-/rcfinder-0.1.9.tgz#f3e80f387ddf9ae80ae30a4100329642eae81115" + dependencies: + lodash.clonedeep "^4.3.2" + +react-collapsible@^2.6.0: + version "2.6.0" + resolved "https://registry.yarnpkg.com/react-collapsible/-/react-collapsible-2.6.0.tgz#8cc18e8b88c46032b21c119833cd6922344636c9" + integrity sha512-r4JSoqh3C3sUIzEr9eXtDC4WPNZsOffO1EeA3ZvyS7JeDIOwmfov9p/380XyewFhqW2B8MHY8IHQ2x1HH5PTxQ== + +react-router-dom@^4.1.2: + version "4.2.2" + resolved "https://registry.yarnpkg.com/react-router-dom/-/react-router-dom-4.2.2.tgz#c8a81df3adc58bba8a76782e946cbd4eae649b8d" + dependencies: + history "^4.7.2" + invariant "^2.2.2" + loose-envify "^1.3.1" + prop-types "^15.5.4" + react-router "^4.2.0" + warning "^3.0.0" + +react-router@^4.1.2, react-router@^4.2.0: + version "4.2.0" + resolved "https://registry.yarnpkg.com/react-router/-/react-router-4.2.0.tgz#61f7b3e3770daeb24062dae3eedef1b054155986" + dependencies: + history "^4.7.2" + hoist-non-react-statics "^2.3.0" + invariant "^2.2.2" + loose-envify "^1.3.1" + path-to-regexp "^1.7.0" + prop-types "^15.5.4" + warning "^3.0.0" + +react-screen-size@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/react-screen-size/-/react-screen-size-1.0.1.tgz#de398c4d7e7989dc14c2a96d19a93b64513afbf6" + +react-timeago@^3.4.3: + version "3.4.3" + resolved "https://registry.yarnpkg.com/react-timeago/-/react-timeago-3.4.3.tgz#eb9061eefb044e4a2b09ce8c99d34645b2dbfa25" + +react-title-component@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/react-title-component/-/react-title-component-1.0.1.tgz#cfd51418f46e8bd42bb01c6cccab51baea0f5700" + +react-transition-group@^1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/react-transition-group/-/react-transition-group-1.2.0.tgz#b51fc921b0c3835a7ef7c571c79fc82c73e9204f" + dependencies: + chain-function "^1.0.0" + dom-helpers "^3.2.0" + loose-envify "^1.3.1" + prop-types "^15.5.6" + warning "^3.0.0" + +read-pkg-up@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/read-pkg-up/-/read-pkg-up-1.0.1.tgz#9d63c13276c065918d57f002a57f40a1b643fb02" + dependencies: + find-up "^1.0.0" + read-pkg "^1.0.0" + +read-pkg-up@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/read-pkg-up/-/read-pkg-up-2.0.0.tgz#6b72a8048984e0c41e79510fd5e9fa99b3b549be" + dependencies: + find-up "^2.0.0" + read-pkg "^2.0.0" + +read-pkg@^1.0.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/read-pkg/-/read-pkg-1.1.0.tgz#f5ffaa5ecd29cb31c0474bca7d756b6bb29e3f28" + dependencies: + load-json-file "^1.0.0" + normalize-package-data "^2.3.2" + path-type "^1.0.0" + +read-pkg@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/read-pkg/-/read-pkg-2.0.0.tgz#8ef1c0623c6a6db0dc6713c4bfac46332b2368f8" + dependencies: + load-json-file "^2.0.0" + normalize-package-data "^2.3.2" + path-type "^2.0.0" + +readable-stream@1.0: + version "1.0.34" + resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-1.0.34.tgz#125820e34bc842d2f2aaafafe4c2916ee32c157c" + dependencies: + core-util-is "~1.0.0" + inherits "~2.0.1" + isarray "0.0.1" + string_decoder "~0.10.x" + +readable-stream@^2.0.1, readable-stream@^2.0.2, readable-stream@^2.0.5, readable-stream@^2.0.6, readable-stream@^2.1.4, readable-stream@^2.2.2, readable-stream@^2.2.6, readable-stream@^2.2.9: + version "2.3.3" + resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-2.3.3.tgz#368f2512d79f9d46fdfc71349ae7878bbc1eb95c" + dependencies: + core-util-is "~1.0.0" + inherits "~2.0.3" + isarray "~1.0.0" + process-nextick-args "~1.0.6" + safe-buffer "~5.1.1" + string_decoder "~1.0.3" + util-deprecate "~1.0.1" + +readdirp@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/readdirp/-/readdirp-2.1.0.tgz#4ed0ad060df3073300c48440373f72d1cc642d78" + dependencies: + graceful-fs "^4.1.2" + minimatch "^3.0.2" + readable-stream "^2.0.2" + set-immediate-shim "^1.0.1" + +redent@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/redent/-/redent-1.0.0.tgz#cf916ab1fd5f1f16dfb20822dd6ec7f730c2afde" + dependencies: + indent-string "^2.1.0" + strip-indent "^1.0.1" + +reduce-css-calc@^1.2.6: + version "1.3.0" + resolved "https://registry.yarnpkg.com/reduce-css-calc/-/reduce-css-calc-1.3.0.tgz#747c914e049614a4c9cfbba629871ad1d2927716" + dependencies: + balanced-match "^0.4.2" + math-expression-evaluator "^1.2.14" + reduce-function-call "^1.0.1" + +reduce-function-call@^1.0.1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/reduce-function-call/-/reduce-function-call-1.0.2.tgz#5a200bf92e0e37751752fe45b0ab330fd4b6be99" + dependencies: + balanced-match "^0.4.2" + +regenerate@^1.2.1: + version "1.3.2" + resolved "https://registry.yarnpkg.com/regenerate/-/regenerate-1.3.2.tgz#d1941c67bad437e1be76433add5b385f95b19260" + +regenerator-runtime@^0.10.5: + version "0.10.5" + resolved "https://registry.yarnpkg.com/regenerator-runtime/-/regenerator-runtime-0.10.5.tgz#336c3efc1220adcedda2c9fab67b5a7955a33658" + +regenerator-runtime@^0.11.0: + version "0.11.0" + resolved "https://registry.yarnpkg.com/regenerator-runtime/-/regenerator-runtime-0.11.0.tgz#7e54fe5b5ccd5d6624ea6255c3473be090b802e1" + +regenerator-transform@^0.10.0: + version "0.10.1" + resolved "https://registry.yarnpkg.com/regenerator-transform/-/regenerator-transform-0.10.1.tgz#1e4996837231da8b7f3cf4114d71b5691a0680dd" + dependencies: + babel-runtime "^6.18.0" + babel-types "^6.19.0" + private "^0.1.6" + +regex-cache@^0.4.2: + version "0.4.4" + resolved "https://registry.yarnpkg.com/regex-cache/-/regex-cache-0.4.4.tgz#75bdc58a2a1496cec48a12835bc54c8d562336dd" + dependencies: + is-equal-shallow "^0.1.3" + +regexpu-core@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/regexpu-core/-/regexpu-core-1.0.0.tgz#86a763f58ee4d7c2f6b102e4764050de7ed90c6b" + dependencies: + regenerate "^1.2.1" + regjsgen "^0.2.0" + regjsparser "^0.1.4" + +regexpu-core@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/regexpu-core/-/regexpu-core-2.0.0.tgz#49d038837b8dcf8bfa5b9a42139938e6ea2ae240" + dependencies: + regenerate "^1.2.1" + regjsgen "^0.2.0" + regjsparser "^0.1.4" + +regjsgen@^0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/regjsgen/-/regjsgen-0.2.0.tgz#6c016adeac554f75823fe37ac05b92d5a4edb1f7" + +regjsparser@^0.1.4: + version "0.1.5" + resolved "https://registry.yarnpkg.com/regjsparser/-/regjsparser-0.1.5.tgz#7ee8f84dc6fa792d3fd0ae228d24bd949ead205c" + dependencies: + jsesc "~0.5.0" + +relateurl@0.2.x: + version "0.2.7" + resolved "https://registry.yarnpkg.com/relateurl/-/relateurl-0.2.7.tgz#54dbf377e51440aca90a4cd274600d3ff2d888a9" + +remove-trailing-separator@^1.0.1: + version "1.1.0" + resolved "https://registry.yarnpkg.com/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz#c24bce2a283adad5bc3f58e0d48249b92379d8ef" + +renderkid@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/renderkid/-/renderkid-2.0.1.tgz#898cabfc8bede4b7b91135a3ffd323e58c0db319" + dependencies: + css-select "^1.1.0" + dom-converter "~0.1" + htmlparser2 "~3.3.0" + strip-ansi "^3.0.0" + utila "~0.3" + +repeat-element@^1.1.2: + version "1.1.2" + resolved "https://registry.yarnpkg.com/repeat-element/-/repeat-element-1.1.2.tgz#ef089a178d1483baae4d93eb98b4f9e4e11d990a" + +repeat-string@^1.5.2: + version "1.6.1" + resolved "https://registry.yarnpkg.com/repeat-string/-/repeat-string-1.6.1.tgz#8dcae470e1c88abc2d600fff4a776286da75e637" + +repeating@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/repeating/-/repeating-2.0.1.tgz#5214c53a926d3552707527fbab415dbc08d06dda" + dependencies: + is-finite "^1.0.0" + +request@^2.72.0, request@^2.79.0, request@^2.81.0: + version "2.81.0" + resolved "https://registry.yarnpkg.com/request/-/request-2.81.0.tgz#c6928946a0e06c5f8d6f8a9333469ffda46298a0" + dependencies: + aws-sign2 "~0.6.0" + aws4 "^1.2.1" + caseless "~0.12.0" + combined-stream "~1.0.5" + extend "~3.0.0" + forever-agent "~0.6.1" + form-data "~2.1.1" + har-validator "~4.2.1" + hawk "~3.1.3" + http-signature "~1.1.0" + is-typedarray "~1.0.0" + isstream "~0.1.2" + json-stringify-safe "~5.0.1" + mime-types "~2.1.7" + oauth-sign "~0.8.1" + performance-now "^0.2.0" + qs "~6.4.0" + safe-buffer "^5.0.1" + stringstream "~0.0.4" + tough-cookie "~2.3.0" + tunnel-agent "^0.6.0" + uuid "^3.0.0" + +require-directory@^2.1.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/require-directory/-/require-directory-2.1.1.tgz#8c64ad5fd30dab1c976e2344ffe7f792a6a6df42" + +require-main-filename@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/require-main-filename/-/require-main-filename-1.0.1.tgz#97f717b69d48784f5f526a6c5aa8ffdda055a4d1" + +require-uncached@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/require-uncached/-/require-uncached-1.0.3.tgz#4e0d56d6c9662fd31e43011c4b95aa49955421d3" + dependencies: + caller-path "^0.1.0" + resolve-from "^1.0.0" + +requires-port@1.0.x, requires-port@1.x.x: + version "1.0.0" + resolved "https://registry.yarnpkg.com/requires-port/-/requires-port-1.0.0.tgz#925d2601d39ac485e091cf0da5c6e694dc3dcaff" + +requizzle@~0.2.1: + version "0.2.1" + resolved "https://registry.yarnpkg.com/requizzle/-/requizzle-0.2.1.tgz#6943c3530c4d9a7e46f1cddd51c158fc670cdbde" + dependencies: + underscore "~1.6.0" + +resolve-from@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-1.0.1.tgz#26cbfe935d1aeeeabb29bc3fe5aeb01e93d44226" + +resolve-pathname@^2.2.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/resolve-pathname/-/resolve-pathname-2.2.0.tgz#7e9ae21ed815fd63ab189adeee64dc831eefa879" + +resolve@1.1.7: + version "1.1.7" + resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.1.7.tgz#203114d82ad2c5ed9e8e0411b3932875e889e97b" + +resolve@^1.2.0, resolve@^1.3.3, resolve@~1.4.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.4.0.tgz#a75be01c53da25d934a98ebd0e4c4a7312f92a86" + dependencies: + path-parse "^1.0.5" + +restore-cursor@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/restore-cursor/-/restore-cursor-2.0.0.tgz#9f7ee287f82fd326d4fd162923d62129eee0dfaf" + dependencies: + onetime "^2.0.0" + signal-exit "^3.0.2" + +resumer@~0.0.0: + version "0.0.0" + resolved "https://registry.yarnpkg.com/resumer/-/resumer-0.0.0.tgz#f1e8f461e4064ba39e82af3cdc2a8c893d076759" + dependencies: + through "~2.3.4" + +right-align@^0.1.1: + version "0.1.3" + resolved "https://registry.yarnpkg.com/right-align/-/right-align-0.1.3.tgz#61339b722fe6a3515689210d24e14c96148613ef" + dependencies: + align-text "^0.1.1" + +rimraf@2, rimraf@^2.2.8, rimraf@^2.5.1, rimraf@^2.6.1: + version "2.6.1" + resolved "https://registry.yarnpkg.com/rimraf/-/rimraf-2.6.1.tgz#c2338ec643df7a1b7fe5c54fa86f57428a55f33d" + dependencies: + glob "^7.0.5" + +ripemd160@^2.0.0, ripemd160@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/ripemd160/-/ripemd160-2.0.1.tgz#0f4584295c53a3628af7e6d79aca21ce57d1c6e7" + dependencies: + hash-base "^2.0.0" + inherits "^2.0.1" + +run-async@^2.2.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/run-async/-/run-async-2.3.0.tgz#0371ab4ae0bdd720d4166d7dfda64ff7a445a6c0" + dependencies: + is-promise "^2.1.0" + +rx-lite-aggregates@^4.0.8: + version "4.0.8" + resolved "https://registry.yarnpkg.com/rx-lite-aggregates/-/rx-lite-aggregates-4.0.8.tgz#753b87a89a11c95467c4ac1626c4efc4e05c67be" + dependencies: + rx-lite "*" + +rx-lite@*, rx-lite@^4.0.8: + version "4.0.8" + resolved "https://registry.yarnpkg.com/rx-lite/-/rx-lite-4.0.8.tgz#0b1e11af8bc44836f04a6407e92da42467b79444" + +safe-buffer@5.1.1, safe-buffer@^5.0.1, safe-buffer@^5.1.0, safe-buffer@^5.1.1, safe-buffer@~5.1.0, safe-buffer@~5.1.1: + version "5.1.1" + resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.1.1.tgz#893312af69b2123def71f57889001671eeb2c853" + +samsam@1.x, samsam@^1.1.3: + version "1.2.1" + resolved "https://registry.yarnpkg.com/samsam/-/samsam-1.2.1.tgz#edd39093a3184370cb859243b2bdf255e7d8ea67" + +sane@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/sane/-/sane-2.0.0.tgz#99cb79f21f4a53a69d4d0cd957c2db04024b8eb2" + dependencies: + anymatch "^1.3.0" + exec-sh "^0.2.0" + fb-watchman "^2.0.0" + minimatch "^3.0.2" + minimist "^1.1.1" + walker "~1.0.5" + watch "~0.10.0" + optionalDependencies: + fsevents "^1.1.1" + +sax@^1.2.1, sax@~1.2.1: + version "1.2.4" + resolved "https://registry.yarnpkg.com/sax/-/sax-1.2.4.tgz#2816234e2378bddc4e5354fab5caa895df7100d9" + +schema-utils@^0.3.0: + version "0.3.0" + resolved "https://registry.yarnpkg.com/schema-utils/-/schema-utils-0.3.0.tgz#f5877222ce3e931edae039f17eb3716e7137f8cf" + dependencies: + ajv "^5.0.0" + +select-hose@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/select-hose/-/select-hose-2.0.0.tgz#625d8658f865af43ec962bfc376a37359a4994ca" + +selfsigned@^1.9.1: + version "1.10.1" + resolved "https://registry.yarnpkg.com/selfsigned/-/selfsigned-1.10.1.tgz#bf8cb7b83256c4551e31347c6311778db99eec52" + dependencies: + node-forge "0.6.33" + +"semver@2 || 3 || 4 || 5", semver@^5.3.0: + version "5.4.1" + resolved "https://registry.yarnpkg.com/semver/-/semver-5.4.1.tgz#e059c09d8571f0540823733433505d3a2f00b18e" + +semver@5.3.0: + version "5.3.0" + resolved "https://registry.yarnpkg.com/semver/-/semver-5.3.0.tgz#9b2ce5d3de02d17c6012ad326aa6b4d0cf54f94f" + +send@0.15.4: + version "0.15.4" + resolved "https://registry.yarnpkg.com/send/-/send-0.15.4.tgz#985faa3e284b0273c793364a35c6737bd93905b9" + dependencies: + debug "2.6.8" + depd "~1.1.1" + destroy "~1.0.4" + encodeurl "~1.0.1" + escape-html "~1.0.3" + etag "~1.8.0" + fresh "0.5.0" + http-errors "~1.6.2" + mime "1.3.4" + ms "2.0.0" + on-finished "~2.3.0" + range-parser "~1.2.0" + statuses "~1.3.1" + +serve-index@^1.7.2: + version "1.9.0" + resolved "https://registry.yarnpkg.com/serve-index/-/serve-index-1.9.0.tgz#d2b280fc560d616ee81b48bf0fa82abed2485ce7" + dependencies: + accepts "~1.3.3" + batch "0.6.1" + debug "2.6.8" + escape-html "~1.0.3" + http-errors "~1.6.1" + mime-types "~2.1.15" + parseurl "~1.3.1" + +serve-static@1.12.4: + version "1.12.4" + resolved "https://registry.yarnpkg.com/serve-static/-/serve-static-1.12.4.tgz#9b6aa98eeb7253c4eedc4c1f6fdbca609901a961" + dependencies: + encodeurl "~1.0.1" + escape-html "~1.0.3" + parseurl "~1.3.1" + send "0.15.4" + +set-blocking@^2.0.0, set-blocking@~2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/set-blocking/-/set-blocking-2.0.0.tgz#045f9782d011ae9a6803ddd382b24392b3d890f7" + +set-immediate-shim@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/set-immediate-shim/-/set-immediate-shim-1.0.1.tgz#4b2b1b27eb808a9f8dcc481a58e5e56f599f3f61" + +setimmediate@^1.0.4, setimmediate@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/setimmediate/-/setimmediate-1.0.5.tgz#290cbb232e306942d7d7ea9b83732ab7856f8285" + +setprototypeof@1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/setprototypeof/-/setprototypeof-1.0.3.tgz#66567e37043eeb4f04d91bd658c0cbefb55b8e04" + +sha.js@^2.4.0, sha.js@^2.4.8: + version "2.4.8" + resolved "https://registry.yarnpkg.com/sha.js/-/sha.js-2.4.8.tgz#37068c2c476b6baf402d14a49c67f597921f634f" + dependencies: + inherits "^2.0.1" + +shebang-command@^1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/shebang-command/-/shebang-command-1.2.0.tgz#44aac65b695b03398968c39f363fee5deafdf1ea" + dependencies: + shebang-regex "^1.0.0" + +shebang-regex@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/shebang-regex/-/shebang-regex-1.0.0.tgz#da42f49740c0b42db2ca9728571cb190c98efea3" + +shellwords@^0.1.0: + version "0.1.1" + resolved "https://registry.yarnpkg.com/shellwords/-/shellwords-0.1.1.tgz#d6b9181c1a48d397324c84871efbcfc73fc0654b" + +signal-exit@^3.0.0, signal-exit@^3.0.2: + version "3.0.2" + resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-3.0.2.tgz#b5fdc08f1287ea1178628e415e25132b73646c6d" + +simulate-event@^1.4.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/simulate-event/-/simulate-event-1.4.0.tgz#7f8a404116280bcbfe26347ddbcbffe5bd2be00e" + dependencies: + xtend "^4.0.1" + +sinon-chai@^2.13.0: + version "2.13.0" + resolved "https://registry.yarnpkg.com/sinon-chai/-/sinon-chai-2.13.0.tgz#b9a42e801c20234bfc2f43b29e6f4f61b60990c4" + +sinon@^3.2.1: + version "3.2.1" + resolved "https://registry.yarnpkg.com/sinon/-/sinon-3.2.1.tgz#d8adabd900730fd497788a027049c64b08be91c2" + dependencies: + diff "^3.1.0" + formatio "1.2.0" + lolex "^2.1.2" + native-promise-only "^0.8.1" + nise "^1.0.1" + path-to-regexp "^1.7.0" + samsam "^1.1.3" + text-encoding "0.6.4" + type-detect "^4.0.0" + +slash@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/slash/-/slash-1.0.0.tgz#c41f2f6c39fc16d1cd17ad4b5d896114ae470d55" + +slice-ansi@0.0.4: + version "0.0.4" + resolved "https://registry.yarnpkg.com/slice-ansi/-/slice-ansi-0.0.4.tgz#edbf8903f66f7ce2f8eafd6ceed65e264c831b35" + +sntp@1.x.x: + version "1.0.9" + resolved "https://registry.yarnpkg.com/sntp/-/sntp-1.0.9.tgz#6541184cc90aeea6c6e7b35e2659082443c66198" + dependencies: + hoek "2.x.x" + +sockjs-client@1.1.4: + version "1.1.4" + resolved "https://registry.yarnpkg.com/sockjs-client/-/sockjs-client-1.1.4.tgz#5babe386b775e4cf14e7520911452654016c8b12" + dependencies: + debug "^2.6.6" + eventsource "0.1.6" + faye-websocket "~0.11.0" + inherits "^2.0.1" + json3 "^3.3.2" + url-parse "^1.1.8" + +sockjs@0.3.18: + version "0.3.18" + resolved "https://registry.yarnpkg.com/sockjs/-/sockjs-0.3.18.tgz#d9b289316ca7df77595ef299e075f0f937eb4207" + dependencies: + faye-websocket "^0.10.0" + uuid "^2.0.2" + +sort-keys@^1.0.0: + version "1.1.2" + resolved "https://registry.yarnpkg.com/sort-keys/-/sort-keys-1.1.2.tgz#441b6d4d346798f1b4e49e8920adfba0e543f9ad" + dependencies: + is-plain-obj "^1.0.0" + +source-list-map@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/source-list-map/-/source-list-map-2.0.0.tgz#aaa47403f7b245a92fbc97ea08f250d6087ed085" + +source-map-support@^0.4.15: + version "0.4.18" + resolved "https://registry.yarnpkg.com/source-map-support/-/source-map-support-0.4.18.tgz#0286a6de8be42641338594e97ccea75f0a2c585f" + dependencies: + source-map "^0.5.6" + +source-map@0.5.x, source-map@^0.5.3, source-map@^0.5.6, source-map@^0.5.7, source-map@~0.5.1, source-map@~0.5.3, source-map@~0.5.6: + version "0.5.7" + resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.5.7.tgz#8a039d2d1021d22d1ea14c80d8ea468ba2ef3fcc" + +source-map@^0.6.1, source-map@~0.6.1: + version "0.6.1" + resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.6.1.tgz#74722af32e9614e9c287a8d0bbde48b5e2f1a263" + integrity sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g== + +spdx-correct@~1.0.0: + version "1.0.2" + resolved "https://registry.yarnpkg.com/spdx-correct/-/spdx-correct-1.0.2.tgz#4b3073d933ff51f3912f03ac5519498a4150db40" + dependencies: + spdx-license-ids "^1.0.2" + +spdx-expression-parse@~1.0.0: + version "1.0.4" + resolved "https://registry.yarnpkg.com/spdx-expression-parse/-/spdx-expression-parse-1.0.4.tgz#9bdf2f20e1f40ed447fbe273266191fced51626c" + +spdx-license-ids@^1.0.2: + version "1.2.2" + resolved "https://registry.yarnpkg.com/spdx-license-ids/-/spdx-license-ids-1.2.2.tgz#c9df7a3424594ade6bd11900d596696dc06bac57" + +spdy-transport@^2.0.18: + version "2.0.20" + resolved "https://registry.yarnpkg.com/spdy-transport/-/spdy-transport-2.0.20.tgz#735e72054c486b2354fe89e702256004a39ace4d" + dependencies: + debug "^2.6.8" + detect-node "^2.0.3" + hpack.js "^2.1.6" + obuf "^1.1.1" + readable-stream "^2.2.9" + safe-buffer "^5.0.1" + wbuf "^1.7.2" + +spdy@^3.4.1: + version "3.4.7" + resolved "https://registry.yarnpkg.com/spdy/-/spdy-3.4.7.tgz#42ff41ece5cc0f99a3a6c28aabb73f5c3b03acbc" + dependencies: + debug "^2.6.8" + handle-thing "^1.2.5" + http-deceiver "^1.2.7" + safe-buffer "^5.0.1" + select-hose "^2.0.0" + spdy-transport "^2.0.18" + +sprintf-js@~1.0.2: + version "1.0.3" + resolved "https://registry.yarnpkg.com/sprintf-js/-/sprintf-js-1.0.3.tgz#04e6926f662895354f3dd015203633b857297e2c" + +sshpk@^1.7.0: + version "1.13.1" + resolved "https://registry.yarnpkg.com/sshpk/-/sshpk-1.13.1.tgz#512df6da6287144316dc4c18fe1cf1d940739be3" + dependencies: + asn1 "~0.2.3" + assert-plus "^1.0.0" + dashdash "^1.12.0" + getpass "^0.1.1" + optionalDependencies: + bcrypt-pbkdf "^1.0.0" + ecc-jsbn "~0.1.1" + jsbn "~0.1.0" + tweetnacl "~0.14.0" + +standalone-react-addons-pure-render-mixin@^0.1.1: + version "0.1.1" + resolved "https://registry.yarnpkg.com/standalone-react-addons-pure-render-mixin/-/standalone-react-addons-pure-render-mixin-0.1.1.tgz#3c7409f4c79c40de9ac72c616cf679a994f37551" + +"statuses@>= 1.3.1 < 2", statuses@~1.3.1: + version "1.3.1" + resolved "https://registry.yarnpkg.com/statuses/-/statuses-1.3.1.tgz#faf51b9eb74aaef3b3acf4ad5f61abf24cb7b93e" + +stream-browserify@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/stream-browserify/-/stream-browserify-2.0.1.tgz#66266ee5f9bdb9940a4e4514cafb43bb71e5c9db" + dependencies: + inherits "~2.0.1" + readable-stream "^2.0.2" + +stream-http@^2.3.1: + version "2.7.2" + resolved "https://registry.yarnpkg.com/stream-http/-/stream-http-2.7.2.tgz#40a050ec8dc3b53b33d9909415c02c0bf1abfbad" + dependencies: + builtin-status-codes "^3.0.0" + inherits "^2.0.1" + readable-stream "^2.2.6" + to-arraybuffer "^1.0.0" + xtend "^4.0.0" + +strict-uri-encode@^1.0.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/strict-uri-encode/-/strict-uri-encode-1.1.0.tgz#279b225df1d582b1f54e65addd4352e18faa0713" + +string-length@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/string-length/-/string-length-2.0.0.tgz#d40dbb686a3ace960c1cffca562bf2c45f8363ed" + dependencies: + astral-regex "^1.0.0" + strip-ansi "^4.0.0" + +string-width@^1.0.1, string-width@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/string-width/-/string-width-1.0.2.tgz#118bdf5b8cdc51a2a7e70d211e07e2b0b9b107d3" + dependencies: + code-point-at "^1.0.0" + is-fullwidth-code-point "^1.0.0" + strip-ansi "^3.0.0" + +string-width@^2.0.0, string-width@^2.1.0: + version "2.1.1" + resolved "https://registry.yarnpkg.com/string-width/-/string-width-2.1.1.tgz#ab93f27a8dc13d28cac815c462143a6d9012ae9e" + dependencies: + is-fullwidth-code-point "^2.0.0" + strip-ansi "^4.0.0" + +string.prototype.trim@~1.1.2: + version "1.1.2" + resolved "https://registry.yarnpkg.com/string.prototype.trim/-/string.prototype.trim-1.1.2.tgz#d04de2c89e137f4d7d206f086b5ed2fae6be8cea" + dependencies: + define-properties "^1.1.2" + es-abstract "^1.5.0" + function-bind "^1.0.2" + +string_decoder@^0.10.25, string_decoder@~0.10.x: + version "0.10.31" + resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-0.10.31.tgz#62e203bc41766c6c28c9fc84301dab1c5310fa94" + +string_decoder@~1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.0.3.tgz#0fc67d7c141825de94282dd536bec6b9bce860ab" + dependencies: + safe-buffer "~5.1.0" + +stringstream@~0.0.4: + version "0.0.5" + resolved "https://registry.yarnpkg.com/stringstream/-/stringstream-0.0.5.tgz#4e484cd4de5a0bbbee18e46307710a8a81621878" + +strip-ansi@^3.0.0, strip-ansi@^3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-3.0.1.tgz#6a385fb8853d952d5ff05d0e8aaf94278dc63dcf" + dependencies: + ansi-regex "^2.0.0" + +strip-ansi@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-4.0.0.tgz#a8479022eb1ac368a871389b635262c505ee368f" + dependencies: + ansi-regex "^3.0.0" + +strip-bom@3.0.0, strip-bom@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/strip-bom/-/strip-bom-3.0.0.tgz#2334c18e9c759f7bdd56fdef7e9ae3d588e68ed3" + +strip-bom@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/strip-bom/-/strip-bom-2.0.0.tgz#6219a85616520491f35788bdbf1447a99c7e6b0e" + dependencies: + is-utf8 "^0.2.0" + +strip-eof@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/strip-eof/-/strip-eof-1.0.0.tgz#bb43ff5598a6eb05d89b59fcd129c983313606bf" + +strip-indent@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/strip-indent/-/strip-indent-1.0.1.tgz#0c7962a6adefa7bbd4ac366460a638552ae1a0a2" + dependencies: + get-stdin "^4.0.1" + +strip-json-comments@^2.0.0, strip-json-comments@~2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-2.0.1.tgz#3c531942e908c2697c0ec344858c286c7ca0a60a" + +style-loader@^0.18.2: + version "0.18.2" + resolved "https://registry.yarnpkg.com/style-loader/-/style-loader-0.18.2.tgz#cc31459afbcd6d80b7220ee54b291a9fd66ff5eb" + dependencies: + loader-utils "^1.0.2" + schema-utils "^0.3.0" + +supports-color@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-2.0.0.tgz#535d045ce6b6363fa40117084629995e9df324c7" + +supports-color@^3.1.1, supports-color@^3.1.2, supports-color@^3.2.3: + version "3.2.3" + resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-3.2.3.tgz#65ac0504b3954171d8a64946b2ae3cbb8a5f54f6" + dependencies: + has-flag "^1.0.0" + +supports-color@^4.0.0, supports-color@^4.2.1, supports-color@^4.4.0: + version "4.4.0" + resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-4.4.0.tgz#883f7ddabc165142b2a61427f3352ded195d1a3e" + dependencies: + has-flag "^2.0.0" + +svgo@^0.7.0: + version "0.7.2" + resolved "https://registry.yarnpkg.com/svgo/-/svgo-0.7.2.tgz#9f5772413952135c6fefbf40afe6a4faa88b4bb5" + dependencies: + coa "~1.0.1" + colors "~1.1.2" + csso "~2.3.1" + js-yaml "~3.7.0" + mkdirp "~0.5.1" + sax "~1.2.1" + whet.extend "~0.9.9" + +symbol-tree@^3.2.1: + version "3.2.2" + resolved "https://registry.yarnpkg.com/symbol-tree/-/symbol-tree-3.2.2.tgz#ae27db38f660a7ae2e1c3b7d1bc290819b8519e6" + +table@^4.0.1: + version "4.0.1" + resolved "https://registry.yarnpkg.com/table/-/table-4.0.1.tgz#a8116c133fac2c61f4a420ab6cdf5c4d61f0e435" + dependencies: + ajv "^4.7.0" + ajv-keywords "^1.0.0" + chalk "^1.1.1" + lodash "^4.0.0" + slice-ansi "0.0.4" + string-width "^2.0.0" + +taffydb@2.6.2: + version "2.6.2" + resolved "https://registry.yarnpkg.com/taffydb/-/taffydb-2.6.2.tgz#7cbcb64b5a141b6a2efc2c5d2c67b4e150b2a268" + +tapable@^0.2.7: + version "0.2.8" + resolved "https://registry.yarnpkg.com/tapable/-/tapable-0.2.8.tgz#99372a5c999bf2df160afc0d74bed4f47948cd22" + +tape@^4.6.3: + version "4.8.0" + resolved "https://registry.yarnpkg.com/tape/-/tape-4.8.0.tgz#f6a9fec41cc50a1de50fa33603ab580991f6068e" + dependencies: + deep-equal "~1.0.1" + defined "~1.0.0" + for-each "~0.3.2" + function-bind "~1.1.0" + glob "~7.1.2" + has "~1.0.1" + inherits "~2.0.3" + minimist "~1.2.0" + object-inspect "~1.3.0" + resolve "~1.4.0" + resumer "~0.0.0" + string.prototype.trim "~1.1.2" + through "~2.3.8" + +tar-pack@^3.4.0: + version "3.4.0" + resolved "https://registry.yarnpkg.com/tar-pack/-/tar-pack-3.4.0.tgz#23be2d7f671a8339376cbdb0b8fe3fdebf317984" + dependencies: + debug "^2.2.0" + fstream "^1.0.10" + fstream-ignore "^1.0.5" + once "^1.3.3" + readable-stream "^2.1.4" + rimraf "^2.5.1" + tar "^2.2.1" + uid-number "^0.0.6" + +tar@^2.2.1: + version "2.2.1" + resolved "https://registry.yarnpkg.com/tar/-/tar-2.2.1.tgz#8e4d2a256c0e2185c6b18ad694aec968b83cb1d1" + dependencies: + block-stream "*" + fstream "^1.0.2" + inherits "2" + +test-exclude@^4.1.1: + version "4.1.1" + resolved "https://registry.yarnpkg.com/test-exclude/-/test-exclude-4.1.1.tgz#4d84964b0966b0087ecc334a2ce002d3d9341e26" + dependencies: + arrify "^1.0.1" + micromatch "^2.3.11" + object-assign "^4.1.0" + read-pkg-up "^1.0.1" + require-main-filename "^1.0.1" + +text-encoding@0.6.4: + version "0.6.4" + resolved "https://registry.yarnpkg.com/text-encoding/-/text-encoding-0.6.4.tgz#e399a982257a276dae428bb92845cb71bdc26d19" + +text-table@~0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/text-table/-/text-table-0.2.0.tgz#7f5ee823ae805207c00af2df4a84ec3fcfa570b4" + +throat@^4.0.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/throat/-/throat-4.1.0.tgz#89037cbc92c56ab18926e6ba4cbb200e15672a6a" + +through@^2.3.6, through@~2.3.4, through@~2.3.8: + version "2.3.8" + resolved "https://registry.yarnpkg.com/through/-/through-2.3.8.tgz#0dd4c9ffaabc357960b1b724115d7e0e86a2e1f5" + +thunky@^0.1.0: + version "0.1.0" + resolved "https://registry.yarnpkg.com/thunky/-/thunky-0.1.0.tgz#bf30146824e2b6e67b0f2d7a4ac8beb26908684e" + +time-stamp@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/time-stamp/-/time-stamp-2.0.0.tgz#95c6a44530e15ba8d6f4a3ecb8c3a3fac46da357" + +timers-browserify@^2.0.2: + version "2.0.4" + resolved "https://registry.yarnpkg.com/timers-browserify/-/timers-browserify-2.0.4.tgz#96ca53f4b794a5e7c0e1bd7cc88a372298fa01e6" + dependencies: + setimmediate "^1.0.4" + +tmp@^0.0.31: + version "0.0.31" + resolved "https://registry.yarnpkg.com/tmp/-/tmp-0.0.31.tgz#8f38ab9438e17315e5dbd8b3657e8bfb277ae4a7" + dependencies: + os-tmpdir "~1.0.1" + +tmpl@1.0.x: + version "1.0.4" + resolved "https://registry.yarnpkg.com/tmpl/-/tmpl-1.0.4.tgz#23640dd7b42d00433911140820e5cf440e521dd1" + +to-arraybuffer@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/to-arraybuffer/-/to-arraybuffer-1.0.1.tgz#7d229b1fcc637e466ca081180836a7aabff83f43" + +to-fast-properties@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/to-fast-properties/-/to-fast-properties-1.0.3.tgz#b83571fa4d8c25b82e231b06e3a3055de4ca1a47" + +toposort@^1.0.0: + version "1.0.3" + resolved "https://registry.yarnpkg.com/toposort/-/toposort-1.0.3.tgz#f02cd8a74bd8be2fc0e98611c3bacb95a171869c" + +tough-cookie@^2.3.2, tough-cookie@~2.3.0: + version "2.3.2" + resolved "https://registry.yarnpkg.com/tough-cookie/-/tough-cookie-2.3.2.tgz#f081f76e4c85720e6c37a5faced737150d84072a" + dependencies: + punycode "^1.4.1" + +tr46@~0.0.3: + version "0.0.3" + resolved "https://registry.yarnpkg.com/tr46/-/tr46-0.0.3.tgz#8184fd347dac9cdc185992f3a6622e14b9d9ab6a" + +trim-newlines@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/trim-newlines/-/trim-newlines-1.0.0.tgz#5887966bb582a4503a41eb524f7d35011815a613" + +trim-right@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/trim-right/-/trim-right-1.0.1.tgz#cb2e1203067e0c8de1f614094b9fe45704ea6003" + +tryit@^1.0.1: + version "1.0.3" + resolved "https://registry.yarnpkg.com/tryit/-/tryit-1.0.3.tgz#393be730a9446fd1ead6da59a014308f36c289cb" + +tty-browserify@0.0.0: + version "0.0.0" + resolved "https://registry.yarnpkg.com/tty-browserify/-/tty-browserify-0.0.0.tgz#a157ba402da24e9bf957f9aa69d524eed42901a6" + +tunnel-agent@^0.6.0: + version "0.6.0" + resolved "https://registry.yarnpkg.com/tunnel-agent/-/tunnel-agent-0.6.0.tgz#27a5dea06b36b04a0a9966774b290868f0fc40fd" + dependencies: + safe-buffer "^5.0.1" + +tweetnacl@^0.14.3, tweetnacl@~0.14.0: + version "0.14.5" + resolved "https://registry.yarnpkg.com/tweetnacl/-/tweetnacl-0.14.5.tgz#5ae68177f192d4456269d108afa93ff8743f4f64" + +type-check@~0.3.2: + version "0.3.2" + resolved "https://registry.yarnpkg.com/type-check/-/type-check-0.3.2.tgz#5884cab512cf1d355e3fb784f30804b2b520db72" + dependencies: + prelude-ls "~1.1.2" + +type-detect@^4.0.0: + version "4.0.3" + resolved "https://registry.yarnpkg.com/type-detect/-/type-detect-4.0.3.tgz#0e3f2670b44099b0b46c284d136a7ef49c74c2ea" + +type-is@~1.6.15: + version "1.6.15" + resolved "https://registry.yarnpkg.com/type-is/-/type-is-1.6.15.tgz#cab10fb4909e441c82842eafe1ad646c81804410" + dependencies: + media-typer "0.3.0" + mime-types "~2.1.15" + +typedarray@^0.0.6: + version "0.0.6" + resolved "https://registry.yarnpkg.com/typedarray/-/typedarray-0.0.6.tgz#867ac74e3864187b1d3d47d996a78ec5c8830777" + +ua-parser-js@^0.7.9: + version "0.7.14" + resolved "https://registry.yarnpkg.com/ua-parser-js/-/ua-parser-js-0.7.14.tgz#110d53fa4c3f326c121292bbeac904d2e03387ca" + +uglify-js@3.0.x: + version "3.0.28" + resolved "https://registry.yarnpkg.com/uglify-js/-/uglify-js-3.0.28.tgz#96b8495f0272944787b5843a1679aa326640d5f7" + dependencies: + commander "~2.11.0" + source-map "~0.5.1" + +uglify-js@^2.8.29: + version "2.8.29" + resolved "https://registry.yarnpkg.com/uglify-js/-/uglify-js-2.8.29.tgz#29c5733148057bb4e1f75df35b7a9cb72e6a59dd" + dependencies: + source-map "~0.5.1" + yargs "~3.10.0" + optionalDependencies: + uglify-to-browserify "~1.0.0" + +uglify-js@^3.1.4: + version "3.6.0" + resolved "https://registry.yarnpkg.com/uglify-js/-/uglify-js-3.6.0.tgz#704681345c53a8b2079fb6cec294b05ead242ff5" + integrity sha512-W+jrUHJr3DXKhrsS7NUVxn3zqMOFn0hL/Ei6v0anCIMoKC93TjcflTagwIHLW7SfMFfiQuktQyFVCFHGUE0+yg== + dependencies: + commander "~2.20.0" + source-map "~0.6.1" + +uglify-to-browserify@~1.0.0: + version "1.0.2" + resolved "https://registry.yarnpkg.com/uglify-to-browserify/-/uglify-to-browserify-1.0.2.tgz#6e0924d6bda6b5afe349e39a6d632850a0f882b7" + +uglifyjs-webpack-plugin@^0.4.6: + version "0.4.6" + resolved "https://registry.yarnpkg.com/uglifyjs-webpack-plugin/-/uglifyjs-webpack-plugin-0.4.6.tgz#b951f4abb6bd617e66f63eb891498e391763e309" + dependencies: + source-map "^0.5.6" + uglify-js "^2.8.29" + webpack-sources "^1.0.1" + +uid-number@^0.0.6: + version "0.0.6" + resolved "https://registry.yarnpkg.com/uid-number/-/uid-number-0.0.6.tgz#0ea10e8035e8eb5b8e4449f06da1c730663baa81" + +underscore-contrib@~0.3.0: + version "0.3.0" + resolved "https://registry.yarnpkg.com/underscore-contrib/-/underscore-contrib-0.3.0.tgz#665b66c24783f8fa2b18c9f8cbb0e2c7d48c26c7" + dependencies: + underscore "1.6.0" + +underscore@1.6.0, underscore@~1.6.0: + version "1.6.0" + resolved "https://registry.yarnpkg.com/underscore/-/underscore-1.6.0.tgz#8b38b10cacdef63337b8b24e4ff86d45aea529a8" + +underscore@~1.8.3: + version "1.8.3" + resolved "https://registry.yarnpkg.com/underscore/-/underscore-1.8.3.tgz#4f3fb53b106e6097fcf9cb4109f2a5e9bdfa5022" + +uniq@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/uniq/-/uniq-1.0.1.tgz#b31c5ae8254844a3a8281541ce2b04b865a734ff" + +uniqid@^4.0.0: + version "4.1.1" + resolved "https://registry.yarnpkg.com/uniqid/-/uniqid-4.1.1.tgz#89220ddf6b751ae52b5f72484863528596bb84c1" + dependencies: + macaddress "^0.2.8" + +uniqs@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/uniqs/-/uniqs-2.0.0.tgz#ffede4b36b25290696e6e165d4a59edb998e6b02" + +unpipe@~1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/unpipe/-/unpipe-1.0.0.tgz#b2bf4ee8514aae6165b4817829d21b2ef49904ec" + +upper-case@^1.1.1: + version "1.1.3" + resolved "https://registry.yarnpkg.com/upper-case/-/upper-case-1.1.3.tgz#f6b4501c2ec4cdd26ba78be7222961de77621598" + +url-loader@^0.5.9: + version "0.5.9" + resolved "https://registry.yarnpkg.com/url-loader/-/url-loader-0.5.9.tgz#cc8fea82c7b906e7777019250869e569e995c295" + dependencies: + loader-utils "^1.0.2" + mime "1.3.x" + +url-parse@1.0.x: + version "1.0.5" + resolved "https://registry.yarnpkg.com/url-parse/-/url-parse-1.0.5.tgz#0854860422afdcfefeb6c965c662d4800169927b" + dependencies: + querystringify "0.0.x" + requires-port "1.0.x" + +url-parse@^1.1.8: + version "1.1.9" + resolved "https://registry.yarnpkg.com/url-parse/-/url-parse-1.1.9.tgz#c67f1d775d51f0a18911dd7b3ffad27bb9e5bd19" + dependencies: + querystringify "~1.0.0" + requires-port "1.0.x" + +url@^0.11.0: + version "0.11.0" + resolved "https://registry.yarnpkg.com/url/-/url-0.11.0.tgz#3838e97cfc60521eb73c525a8e55bfdd9e2e28f1" + dependencies: + punycode "1.3.2" + querystring "0.2.0" + +util-deprecate@~1.0.1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf" + +util@0.10.3, util@^0.10.3: + version "0.10.3" + resolved "https://registry.yarnpkg.com/util/-/util-0.10.3.tgz#7afb1afe50805246489e3db7fe0ed379336ac0f9" + dependencies: + inherits "2.0.1" + +utila@~0.3: + version "0.3.3" + resolved "https://registry.yarnpkg.com/utila/-/utila-0.3.3.tgz#d7e8e7d7e309107092b05f8d9688824d633a4226" + +utila@~0.4: + version "0.4.0" + resolved "https://registry.yarnpkg.com/utila/-/utila-0.4.0.tgz#8a16a05d445657a3aea5eecc5b12a4fa5379772c" + +utils-merge@1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/utils-merge/-/utils-merge-1.0.0.tgz#0294fb922bb9375153541c4f7096231f287c8af8" + +uuid@^2.0.2: + version "2.0.3" + resolved "https://registry.yarnpkg.com/uuid/-/uuid-2.0.3.tgz#67e2e863797215530dff318e5bf9dcebfd47b21a" + +uuid@^3.0.0, uuid@^3.0.1: + version "3.1.0" + resolved "https://registry.yarnpkg.com/uuid/-/uuid-3.1.0.tgz#3dd3d3e790abc24d7b0d3a034ffababe28ebbc04" + +validate-npm-package-license@^3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/validate-npm-package-license/-/validate-npm-package-license-3.0.1.tgz#2804babe712ad3379459acfbe24746ab2c303fbc" + dependencies: + spdx-correct "~1.0.0" + spdx-expression-parse "~1.0.0" + +value-equal@^0.4.0: + version "0.4.0" + resolved "https://registry.yarnpkg.com/value-equal/-/value-equal-0.4.0.tgz#c5bdd2f54ee093c04839d71ce2e4758a6890abc7" + +vary@~1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/vary/-/vary-1.1.1.tgz#67535ebb694c1d52257457984665323f587e8d37" + +vendors@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/vendors/-/vendors-1.0.1.tgz#37ad73c8ee417fb3d580e785312307d274847f22" + +verror@1.10.0: + version "1.10.0" + resolved "https://registry.yarnpkg.com/verror/-/verror-1.10.0.tgz#3a105ca17053af55d6e270c1f8288682e18da400" + dependencies: + assert-plus "^1.0.0" + core-util-is "1.0.2" + extsprintf "^1.2.0" + +vm-browserify@0.0.4: + version "0.0.4" + resolved "https://registry.yarnpkg.com/vm-browserify/-/vm-browserify-0.0.4.tgz#5d7ea45bbef9e4a6ff65f95438e0a87c357d5a73" + dependencies: + indexof "0.0.1" + +walker@~1.0.5: + version "1.0.7" + resolved "https://registry.yarnpkg.com/walker/-/walker-1.0.7.tgz#2f7f9b8fd10d677262b18a884e28d19618e028fb" + dependencies: + makeerror "1.0.x" + +warning@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/warning/-/warning-3.0.0.tgz#32e5377cb572de4ab04753bdf8821c01ed605b7c" + dependencies: + loose-envify "^1.0.0" + +watch@~0.10.0: + version "0.10.0" + resolved "https://registry.yarnpkg.com/watch/-/watch-0.10.0.tgz#77798b2da0f9910d595f1ace5b0c2258521f21dc" + +watchpack@^1.4.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/watchpack/-/watchpack-1.4.0.tgz#4a1472bcbb952bd0a9bb4036801f954dfb39faac" + dependencies: + async "^2.1.2" + chokidar "^1.7.0" + graceful-fs "^4.1.2" + +wbuf@^1.1.0, wbuf@^1.7.2: + version "1.7.2" + resolved "https://registry.yarnpkg.com/wbuf/-/wbuf-1.7.2.tgz#d697b99f1f59512df2751be42769c1580b5801fe" + dependencies: + minimalistic-assert "^1.0.0" + +webidl-conversions@^3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/webidl-conversions/-/webidl-conversions-3.0.1.tgz#24534275e2a7bc6be7bc86611cc16ae0a5654871" + +webidl-conversions@^4.0.0: + version "4.0.2" + resolved "https://registry.yarnpkg.com/webidl-conversions/-/webidl-conversions-4.0.2.tgz#a855980b1f0b6b359ba1d5d9fb39ae941faa63ad" + +webpack-dev-middleware@^1.11.0: + version "1.12.0" + resolved "https://registry.yarnpkg.com/webpack-dev-middleware/-/webpack-dev-middleware-1.12.0.tgz#d34efefb2edda7e1d3b5dbe07289513219651709" + dependencies: + memory-fs "~0.4.1" + mime "^1.3.4" + path-is-absolute "^1.0.0" + range-parser "^1.0.3" + time-stamp "^2.0.0" + +webpack-dev-server@^2.6.1: + version "2.7.1" + resolved "https://registry.yarnpkg.com/webpack-dev-server/-/webpack-dev-server-2.7.1.tgz#21580f5a08cd065c71144cf6f61c345bca59a8b8" + dependencies: + ansi-html "0.0.7" + bonjour "^3.5.0" + chokidar "^1.6.0" + compression "^1.5.2" + connect-history-api-fallback "^1.3.0" + del "^3.0.0" + express "^4.13.3" + html-entities "^1.2.0" + http-proxy-middleware "~0.17.4" + internal-ip "^1.2.0" + ip "^1.1.5" + loglevel "^1.4.1" + opn "4.0.2" + portfinder "^1.0.9" + selfsigned "^1.9.1" + serve-index "^1.7.2" + sockjs "0.3.18" + sockjs-client "1.1.4" + spdy "^3.4.1" + strip-ansi "^3.0.0" + supports-color "^3.1.1" + webpack-dev-middleware "^1.11.0" + yargs "^6.0.0" + +webpack-sources@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/webpack-sources/-/webpack-sources-1.0.1.tgz#c7356436a4d13123be2e2426a05d1dad9cbe65cf" + dependencies: + source-list-map "^2.0.0" + source-map "~0.5.3" + +webpack@^3.4.1: + version "3.5.6" + resolved "https://registry.yarnpkg.com/webpack/-/webpack-3.5.6.tgz#a492fb6c1ed7f573816f90e00c8fbb5a20cc5c36" + dependencies: + acorn "^5.0.0" + acorn-dynamic-import "^2.0.0" + ajv "^5.1.5" + ajv-keywords "^2.0.0" + async "^2.1.2" + enhanced-resolve "^3.4.0" + escope "^3.6.0" + interpret "^1.0.0" + json-loader "^0.5.4" + json5 "^0.5.1" + loader-runner "^2.3.0" + loader-utils "^1.1.0" + memory-fs "~0.4.1" + mkdirp "~0.5.0" + node-libs-browser "^2.0.0" + source-map "^0.5.3" + supports-color "^4.2.1" + tapable "^0.2.7" + uglifyjs-webpack-plugin "^0.4.6" + watchpack "^1.4.0" + webpack-sources "^1.0.1" + yargs "^8.0.2" + +websocket-driver@>=0.5.1: + version "0.6.5" + resolved "https://registry.yarnpkg.com/websocket-driver/-/websocket-driver-0.6.5.tgz#5cb2556ceb85f4373c6d8238aa691c8454e13a36" + dependencies: + websocket-extensions ">=0.1.1" + +websocket-extensions@>=0.1.1: + version "0.1.2" + resolved "https://registry.yarnpkg.com/websocket-extensions/-/websocket-extensions-0.1.2.tgz#0e18781de629a18308ce1481650f67ffa2693a5d" + +whatwg-encoding@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/whatwg-encoding/-/whatwg-encoding-1.0.1.tgz#3c6c451a198ee7aec55b1ec61d0920c67801a5f4" + dependencies: + iconv-lite "0.4.13" + +whatwg-fetch@>=0.10.0: + version "2.0.3" + resolved "https://registry.yarnpkg.com/whatwg-fetch/-/whatwg-fetch-2.0.3.tgz#9c84ec2dcf68187ff00bc64e1274b442176e1c84" + +whatwg-url@^4.3.0: + version "4.8.0" + resolved "https://registry.yarnpkg.com/whatwg-url/-/whatwg-url-4.8.0.tgz#d2981aa9148c1e00a41c5a6131166ab4683bbcc0" + dependencies: + tr46 "~0.0.3" + webidl-conversions "^3.0.0" + +whet.extend@~0.9.9: + version "0.9.9" + resolved "https://registry.yarnpkg.com/whet.extend/-/whet.extend-0.9.9.tgz#f877d5bf648c97e5aa542fadc16d6a259b9c11a1" + +which-module@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/which-module/-/which-module-1.0.0.tgz#bba63ca861948994ff307736089e3b96026c2a4f" + +which-module@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/which-module/-/which-module-2.0.0.tgz#d9ef07dce77b9902b8a3a8fa4b31c3e3f7e6e87a" + +which@^1.2.12, which@^1.2.9: + version "1.3.0" + resolved "https://registry.yarnpkg.com/which/-/which-1.3.0.tgz#ff04bdfc010ee547d780bec38e1ac1c2777d253a" + dependencies: + isexe "^2.0.0" + +wide-align@^1.1.0: + version "1.1.2" + resolved "https://registry.yarnpkg.com/wide-align/-/wide-align-1.1.2.tgz#571e0f1b0604636ebc0dfc21b0339bbe31341710" + dependencies: + string-width "^1.0.2" + +window-size@0.1.0: + version "0.1.0" + resolved "https://registry.yarnpkg.com/window-size/-/window-size-0.1.0.tgz#5438cd2ea93b202efa3a19fe8887aee7c94f9c9d" + +wordwrap@0.0.2: + version "0.0.2" + resolved "https://registry.yarnpkg.com/wordwrap/-/wordwrap-0.0.2.tgz#b79669bb42ecb409f83d583cad52ca17eaa1643f" + +wordwrap@~0.0.2: + version "0.0.3" + resolved "https://registry.yarnpkg.com/wordwrap/-/wordwrap-0.0.3.tgz#a3d5da6cd5c0bc0008d37234bbaf1bed63059107" + integrity sha1-o9XabNXAvAAI03I0u68b7WMFkQc= + +wordwrap@~1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/wordwrap/-/wordwrap-1.0.0.tgz#27584810891456a4171c8d0226441ade90cbcaeb" + +worker-farm@^1.3.1: + version "1.5.0" + resolved "https://registry.yarnpkg.com/worker-farm/-/worker-farm-1.5.0.tgz#adfdf0cd40581465ed0a1f648f9735722afd5c8d" + dependencies: + errno "^0.1.4" + xtend "^4.0.1" + +wrap-ansi@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-2.1.0.tgz#d8fc3d284dd05794fe84973caecdd1cf824fdd85" + dependencies: + string-width "^1.0.1" + strip-ansi "^3.0.1" + +wrappy@1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f" + +write-file-atomic@^2.1.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/write-file-atomic/-/write-file-atomic-2.3.0.tgz#1ff61575c2e2a4e8e510d6fa4e243cce183999ab" + dependencies: + graceful-fs "^4.1.11" + imurmurhash "^0.1.4" + signal-exit "^3.0.2" + +write@^0.2.1: + version "0.2.1" + resolved "https://registry.yarnpkg.com/write/-/write-0.2.1.tgz#5fc03828e264cea3fe91455476f7a3c566cb0757" + dependencies: + mkdirp "^0.5.1" + +xml-char-classes@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/xml-char-classes/-/xml-char-classes-1.0.0.tgz#64657848a20ffc5df583a42ad8a277b4512bbc4d" + +xml-name-validator@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/xml-name-validator/-/xml-name-validator-2.0.1.tgz#4d8b8f1eccd3419aa362061becef515e1e559635" + +xmlcreate@^1.0.1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/xmlcreate/-/xmlcreate-1.0.2.tgz#fa6bf762a60a413fb3dd8f4b03c5b269238d308f" + +xtend@^4.0.0, xtend@^4.0.1: + version "4.0.1" + resolved "https://registry.yarnpkg.com/xtend/-/xtend-4.0.1.tgz#a5c6d532be656e23db820efb943a1f04998d63af" + +y18n@^3.2.1: + version "3.2.1" + resolved "https://registry.yarnpkg.com/y18n/-/y18n-3.2.1.tgz#6d15fba884c08679c0d77e88e7759e811e07fa41" + +yallist@^2.1.2: + version "2.1.2" + resolved "https://registry.yarnpkg.com/yallist/-/yallist-2.1.2.tgz#1c11f9218f076089a47dd512f93c6699a6a81d52" + +yargs-parser@^4.2.0: + version "4.2.1" + resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-4.2.1.tgz#29cceac0dc4f03c6c87b4a9f217dd18c9f74871c" + dependencies: + camelcase "^3.0.0" + +yargs-parser@^7.0.0: + version "7.0.0" + resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-7.0.0.tgz#8d0ac42f16ea55debd332caf4c4038b3e3f5dfd9" + dependencies: + camelcase "^4.1.0" + +yargs@^6.0.0: + version "6.6.0" + resolved "https://registry.yarnpkg.com/yargs/-/yargs-6.6.0.tgz#782ec21ef403345f830a808ca3d513af56065208" + dependencies: + camelcase "^3.0.0" + cliui "^3.2.0" + decamelize "^1.1.1" + get-caller-file "^1.0.1" + os-locale "^1.4.0" + read-pkg-up "^1.0.1" + require-directory "^2.1.1" + require-main-filename "^1.0.1" + set-blocking "^2.0.0" + string-width "^1.0.2" + which-module "^1.0.0" + y18n "^3.2.1" + yargs-parser "^4.2.0" + +yargs@^8.0.2: + version "8.0.2" + resolved "https://registry.yarnpkg.com/yargs/-/yargs-8.0.2.tgz#6299a9055b1cefc969ff7e79c1d918dceb22c360" + dependencies: + camelcase "^4.1.0" + cliui "^3.2.0" + decamelize "^1.1.1" + get-caller-file "^1.0.1" + os-locale "^2.0.0" + read-pkg-up "^2.0.0" + require-directory "^2.1.1" + require-main-filename "^1.0.1" + set-blocking "^2.0.0" + string-width "^2.0.0" + which-module "^2.0.0" + y18n "^3.2.1" + yargs-parser "^7.0.0" + +yargs@^9.0.0: + version "9.0.0" + resolved "https://registry.yarnpkg.com/yargs/-/yargs-9.0.0.tgz#efe5b1ad3f94bdc20423411b90628eeec0b25f3c" + dependencies: + camelcase "^4.1.0" + cliui "^3.2.0" + decamelize "^1.1.1" + get-caller-file "^1.0.1" + os-locale "^2.0.0" + read-pkg-up "^2.0.0" + require-directory "^2.1.1" + require-main-filename "^1.0.1" + set-blocking "^2.0.0" + string-width "^2.0.0" + which-module "^2.0.0" + y18n "^3.2.1" + yargs-parser "^7.0.0" + +yargs@~3.10.0: + version "3.10.0" + resolved "https://registry.yarnpkg.com/yargs/-/yargs-3.10.0.tgz#f7ee7bd857dd7c1d2d38c0e74efbd681d1431fd1" + dependencies: + camelcase "^1.0.2" + cliui "^2.1.0" + decamelize "^1.0.0" + window-size "0.1.0" + +yarn@^1.6.0: + version "1.6.0" + resolved "https://registry.yarnpkg.com/yarn/-/yarn-1.6.0.tgz#9cec6f7986dc237d39ec705ce74d95155fe55d4b"