├── .gitignore ├── Gruntfile.js ├── LICENSE ├── README.md ├── dist ├── README.md ├── dashboards │ └── demodashboard.json ├── datasource.js ├── datasource.js.map ├── module.js ├── module.js.map ├── partials │ ├── config.html │ └── query.editor.html ├── plugin.json ├── query_ctrl.js ├── query_ctrl.js.map ├── stream_handler.js ├── stream_handler.js.map └── vendor │ ├── ndjson.js │ └── rxjs.umd.min.js ├── livedata.gif ├── package.json ├── server.go ├── src ├── dashboards │ └── demodashboard.json ├── datasource.js ├── module.js ├── partials │ ├── config.html │ └── query.editor.html ├── plugin.json ├── query_ctrl.js ├── stream_handler.js └── vendor │ ├── ndjson.js │ └── rxjs.umd.min.js └── yarn.lock /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | npm-debug.log 3 | coverage/ 4 | .aws-config.json 5 | awsconfig 6 | /emails/dist 7 | /public_gen 8 | /tmp 9 | vendor/phantomjs/phantomjs 10 | 11 | docs/AWS_S3_BUCKET 12 | docs/GIT_BRANCH 13 | docs/VERSION 14 | docs/GITCOMMIT 15 | docs/changed-files 16 | docs/changed-files 17 | 18 | # locally required config files 19 | public/css/*.min.css 20 | 21 | # Editor junk 22 | *.sublime-workspace 23 | *.swp 24 | .idea/ 25 | *.iml 26 | 27 | /data/* 28 | /bin/* 29 | 30 | conf/custom.ini 31 | fig.yml 32 | profile.cov 33 | grafana 34 | .notouch 35 | 36 | # Test artifacts 37 | /dist/test/ 38 | -------------------------------------------------------------------------------- /Gruntfile.js: -------------------------------------------------------------------------------- 1 | module.exports = function(grunt) { 2 | require('load-grunt-tasks')(grunt); 3 | 4 | grunt.loadNpmTasks('grunt-execute'); 5 | grunt.loadNpmTasks('grunt-contrib-clean'); 6 | 7 | grunt.initConfig({ 8 | clean: ['dist'], 9 | 10 | copy: { 11 | src_to_dist: { 12 | cwd: 'src', 13 | expand: true, 14 | src: ['**/*', '!**/*.js', '!**/*.scss', 'vendor/**'], 15 | dest: 'dist', 16 | }, 17 | pluginDef: { 18 | expand: true, 19 | src: ['README.md'], 20 | dest: 'dist', 21 | }, 22 | dashboards: { 23 | expand: true, 24 | cwd: 'src/dashboards', 25 | src: ['*.json'], 26 | dest: 'dist/dashboards/', 27 | }, 28 | }, 29 | 30 | watch: { 31 | rebuild_all: { 32 | files: ['src/**/*'], 33 | tasks: ['default'], 34 | options: { spawn: false }, 35 | }, 36 | }, 37 | 38 | babel: { 39 | options: { 40 | sourceMap: true, 41 | presets: ['env'], 42 | plugins: ['transform-object-rest-spread'], 43 | }, 44 | dist: { 45 | files: [ 46 | { 47 | cwd: 'src', 48 | expand: true, 49 | src: ['**/*.js', '!vendor/**'], 50 | dest: 'dist', 51 | ext: '.js', 52 | }, 53 | ], 54 | }, 55 | }, 56 | }); 57 | 58 | grunt.registerTask('default', ['clean', 'copy:src_to_dist', 'copy:pluginDef', 'copy:dashboards', 'babel']); 59 | }; 60 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | The MIT License (MIT) 2 | 3 | Copyright (c) 2016 Grafana 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | > As of Grafana v6.4, the hacks in this repo are no longer supported/necessary. Grafana v6.4 added native streaming support after positive community feedback. Please refer the offical streaming examples in the [testdata-datasource](https://github.com/grafana/grafana/tree/master/public/app/plugins/datasource/testdata) or alternatively, the [loki-datasource](https://github.com/grafana/grafana/tree/master/public/app/plugins/datasource/loki). 2 | 3 | ## Simple Streaming Datasource 4 | 5 | Inspired by the live-streaming [talk](https://www.youtube.com/watch?v=bPrDTvlNIj8&feature=youtu.be&t=4754) at Grafanacon 2019, this is a living example of streaming within a datasource including a reference server that streams random data. 6 | 7 | ![live streaming dashboard animation](/livedata.gif "live streaming dashboard") 8 | 9 | This example fixes some of the shortcomings mentioned in the talk. 10 | 11 | * Panel repaints are now throttled via RXJS throttling. You can tune the frequency to your use case. 12 | * Streams are cancelled/restarted when queries change. 13 | 14 | ## Motivation 15 | Streaming data is a great way to reduce pressure on your metric backend/network. Rather than using a 5s dashboard refresh (which requests duplicate points over and over again), stream new data as its avaiable! 16 | 17 | ## Running the example 18 | 19 | Install as you would any grafana datasource. Then, run `go run server.go`. This will spawn a server at `http://localhost:8080` that this datasource reads data from. Also included is a demo dashboard. 20 | 21 | By default, the server will stream data at 50ms and the datasource plugin will throttle repaints to 100ms. 22 | 23 | ## Limitations 24 | 25 | * Clicking the dashboard refresh, changing timerange, and some other report interactions do not cause the panel to "refresh". Here's a discussion about it https://github.com/grafana/grafana/issues/15760 26 | * Backend server doesn't support providing a start/end time (it always starts streaming from time.Now()) for simplicity. 27 | * Plugin makes no attempt to order the data. If its streamed in unordered, some backwards lines will be drawn. 28 | * Adding more than one query in the datasource is currently unsupported, but could be easily added with some more state management. You'll likely want to multiplex all queries through the same stream (rather than open a stream per query) if your datasource can support it. 29 | -------------------------------------------------------------------------------- /dist/README.md: -------------------------------------------------------------------------------- 1 | ## Simple Streaming Datasource 2 | 3 | ### End-to-end Grafana live streaming implementation 4 | 5 | Inspired by the live-streaming [talk](https://www.youtube.com/watch?v=bPrDTvlNIj8&feature=youtu.be&t=4754) at Grafanacon 2019, this is a living example of streaming within a datasource- including a reference server that streams random data. 6 | 7 | ![live streaming dashboard animation](/livedata.gif "live streaming dashboard") 8 | 9 | This example fixes some of the short commings mentioned mentioned in the talk 10 | 11 | * Panel repaints are now throttled via RXJS throttling. You can tune the frequency to your use case. 12 | * Streams are cancelled/restarted when queries change 13 | 14 | ## Running the example 15 | 16 | Install as you would any grafana datasource. Then, run `go run server.go`. This will spawn a server at `http://localhost:8080` that this datasource reads data from. Also included is a demo dashboard. 17 | 18 | By default, the sever will stream data at 50ms, and the datasource plugin will throttle repaints to 100ms. 19 | 20 | ## Limitations 21 | 22 | * Clicking the dashboard refresh, changing timerange, and some other report interactions do not cause the panel to "refresh". 23 | * Backend server doesn't support providing a start/end time (it always starts streaming from time.Now()) for simplicity. 24 | * Plugin makes no attempt to order the data. If its streamed in unordered, some backwards lines will be drawn. 25 | * Adding more than one query in the datasource is currently unsupported, but could be easily added with some more state management. You'll likely want to multiplex all queries through the same stream (rather than open a stream per query) if your datasource can support it. 26 | -------------------------------------------------------------------------------- /dist/dashboards/demodashboard.json: -------------------------------------------------------------------------------- 1 | { 2 | "annotations": { 3 | "list": [ 4 | { 5 | "builtIn": 1, 6 | "datasource": "-- Grafana --", 7 | "enable": true, 8 | "hide": true, 9 | "iconColor": "rgba(0, 211, 255, 1)", 10 | "name": "Annotations & Alerts", 11 | "type": "dashboard" 12 | } 13 | ] 14 | }, 15 | "editable": true, 16 | "gnetId": null, 17 | "graphTooltip": 0, 18 | "id": 2, 19 | "links": [], 20 | "panels": [ 21 | { 22 | "cacheTimeout": null, 23 | "colorBackground": false, 24 | "colorValue": false, 25 | "colors": ["#299c46", "rgba(237, 129, 40, 0.89)", "#d44a3a"], 26 | "datasource": "Simple Streaming Datasource", 27 | "decimals": 2, 28 | "format": "none", 29 | "gauge": { 30 | "maxValue": 10, 31 | "minValue": 0, 32 | "show": true, 33 | "thresholdLabels": false, 34 | "thresholdMarkers": true 35 | }, 36 | "gridPos": { 37 | "h": 7, 38 | "w": 8, 39 | "x": 0, 40 | "y": 0 41 | }, 42 | "id": 4, 43 | "interval": null, 44 | "links": [], 45 | "mappingType": 1, 46 | "mappingTypes": [ 47 | { 48 | "name": "value to text", 49 | "value": 1 50 | }, 51 | { 52 | "name": "range to text", 53 | "value": 2 54 | } 55 | ], 56 | "maxDataPoints": 100, 57 | "nullPointMode": "connected", 58 | "nullText": null, 59 | "postfix": "", 60 | "postfixFontSize": "50%", 61 | "prefix": "", 62 | "prefixFontSize": "50%", 63 | "rangeMaps": [ 64 | { 65 | "from": "null", 66 | "text": "N/A", 67 | "to": "null" 68 | } 69 | ], 70 | "sparkline": { 71 | "fillColor": "rgba(31, 118, 189, 0.18)", 72 | "full": false, 73 | "lineColor": "rgb(31, 120, 193)", 74 | "show": false 75 | }, 76 | "tableColumn": "", 77 | "targets": [ 78 | { 79 | "numSeries": 1, 80 | "refId": "A" 81 | } 82 | ], 83 | "thresholds": "", 84 | "title": "Panel Title", 85 | "type": "singlestat", 86 | "valueFontSize": "80%", 87 | "valueMaps": [ 88 | { 89 | "op": "=", 90 | "text": "N/A", 91 | "value": "null" 92 | } 93 | ], 94 | "valueName": "avg" 95 | }, 96 | { 97 | "aliasColors": {}, 98 | "bars": false, 99 | "dashLength": 10, 100 | "dashes": false, 101 | "datasource": "Simple Streaming Datasource", 102 | "fill": 1, 103 | "gridPos": { 104 | "h": 7, 105 | "w": 16, 106 | "x": 8, 107 | "y": 0 108 | }, 109 | "id": 2, 110 | "legend": { 111 | "avg": false, 112 | "current": false, 113 | "max": false, 114 | "min": false, 115 | "show": true, 116 | "total": false, 117 | "values": false 118 | }, 119 | "lines": true, 120 | "linewidth": 1, 121 | "nullPointMode": "null", 122 | "percentage": false, 123 | "pointradius": 5, 124 | "points": false, 125 | "renderer": "flot", 126 | "seriesOverrides": [], 127 | "spaceLength": 10, 128 | "stack": false, 129 | "steppedLine": false, 130 | "targets": [ 131 | { 132 | "numSeries": 4, 133 | "refId": "A", 134 | "target": "select metrich", 135 | "type": "timeserie" 136 | } 137 | ], 138 | "thresholds": [], 139 | "timeFrom": null, 140 | "timeRegions": [], 141 | "timeShift": null, 142 | "title": "Panel Title", 143 | "tooltip": { 144 | "shared": true, 145 | "sort": 0, 146 | "value_type": "individual" 147 | }, 148 | "type": "graph", 149 | "xaxis": { 150 | "buckets": null, 151 | "mode": "time", 152 | "name": null, 153 | "show": true, 154 | "values": [] 155 | }, 156 | "yaxes": [ 157 | { 158 | "format": "short", 159 | "label": null, 160 | "logBase": 1, 161 | "max": null, 162 | "min": null, 163 | "show": true 164 | }, 165 | { 166 | "format": "short", 167 | "label": null, 168 | "logBase": 1, 169 | "max": null, 170 | "min": null, 171 | "show": true 172 | } 173 | ], 174 | "yaxis": { 175 | "align": false, 176 | "alignLevel": null 177 | } 178 | } 179 | ], 180 | "schemaVersion": 16, 181 | "style": "dark", 182 | "tags": [], 183 | "templating": { 184 | "list": [] 185 | }, 186 | "time": { 187 | "from": "now-30s", 188 | "to": "now" 189 | }, 190 | "timepicker": { 191 | "refresh_intervals": ["5s", "10s", "30s", "1m", "5m", "15m", "30m", "1h", "2h", "1d"], 192 | "time_options": ["5m", "15m", "1h", "6h", "12h", "24h", "2d", "7d", "30d"] 193 | }, 194 | "timezone": "", 195 | "title": "Streaming Demo Dashboard", 196 | "uid": "kfYJBgCmk", 197 | "version": 2 198 | } 199 | -------------------------------------------------------------------------------- /dist/datasource.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | Object.defineProperty(exports, "__esModule", { 4 | value: true 5 | }); 6 | exports.StreamingDatasource = undefined; 7 | 8 | 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; }; }(); 9 | 10 | var _stream_handler = require('./stream_handler'); 11 | 12 | function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } 13 | 14 | var StreamingDatasource = exports.StreamingDatasource = function () { 15 | function StreamingDatasource(instanceSettings, $q) { 16 | _classCallCheck(this, StreamingDatasource); 17 | 18 | this.url = instanceSettings.url; 19 | this.q = $q; 20 | this.headers = { 'Content-Type': 'application/json' }; 21 | 22 | // Keep a list of open streams so we can cancel them when querys/timerange change 23 | this.openStreams = []; 24 | } 25 | 26 | _createClass(StreamingDatasource, [{ 27 | key: 'query', 28 | value: function query(options) { 29 | var stream = this.openStreams[options.panelId]; 30 | if (stream) { 31 | return Promise.resolve(stream); 32 | } 33 | 34 | stream = new _stream_handler.StreamHandler(options, this); 35 | this.openStreams[options.panelId] = stream; 36 | stream.open(); 37 | 38 | return Promise.resolve(stream); 39 | } 40 | }, { 41 | key: 'closeStream', 42 | value: function closeStream(panelId) { 43 | var stream = this.openStreams[panelId]; 44 | if (stream) { 45 | stream.close(); 46 | delete this.openStreams[panelId]; 47 | } 48 | } 49 | }]); 50 | 51 | return StreamingDatasource; 52 | }(); 53 | //# sourceMappingURL=datasource.js.map 54 | -------------------------------------------------------------------------------- /dist/datasource.js.map: -------------------------------------------------------------------------------- 1 | {"version":3,"sources":["../src/datasource.js"],"names":["StreamingDatasource","instanceSettings","$q","url","q","headers","openStreams","options","stream","panelId","Promise","resolve","StreamHandler","open","close"],"mappings":";;;;;;;;;AAAA;;;;IAEaA,mB,WAAAA,mB;AACX,+BAAYC,gBAAZ,EAA8BC,EAA9B,EAAkC;AAAA;;AAChC,SAAKC,GAAL,GAAWF,iBAAiBE,GAA5B;AACA,SAAKC,CAAL,GAASF,EAAT;AACA,SAAKG,OAAL,GAAe,EAAE,gBAAgB,kBAAlB,EAAf;;AAEA;AACA,SAAKC,WAAL,GAAmB,EAAnB;AACD;;;;0BAEKC,O,EAAS;AACb,UAAIC,SAAS,KAAKF,WAAL,CAAiBC,QAAQE,OAAzB,CAAb;AACA,UAAID,MAAJ,EAAY;AACV,eAAOE,QAAQC,OAAR,CAAgBH,MAAhB,CAAP;AACD;;AAEDA,eAAS,IAAII,6BAAJ,CAAkBL,OAAlB,EAA2B,IAA3B,CAAT;AACA,WAAKD,WAAL,CAAiBC,QAAQE,OAAzB,IAAoCD,MAApC;AACAA,aAAOK,IAAP;;AAEA,aAAOH,QAAQC,OAAR,CAAgBH,MAAhB,CAAP;AACD;;;gCAEWC,O,EAAS;AACnB,UAAID,SAAS,KAAKF,WAAL,CAAiBG,OAAjB,CAAb;AACA,UAAID,MAAJ,EAAY;AACVA,eAAOM,KAAP;AACA,eAAO,KAAKR,WAAL,CAAiBG,OAAjB,CAAP;AACD;AACF","file":"datasource.js","sourcesContent":["import { StreamHandler } from './stream_handler';\n\nexport class StreamingDatasource {\n constructor(instanceSettings, $q) {\n this.url = instanceSettings.url;\n this.q = $q;\n this.headers = { 'Content-Type': 'application/json' };\n\n // Keep a list of open streams so we can cancel them when querys/timerange change\n this.openStreams = [];\n }\n\n query(options) {\n var stream = this.openStreams[options.panelId];\n if (stream) {\n return Promise.resolve(stream);\n }\n\n stream = new StreamHandler(options, this);\n this.openStreams[options.panelId] = stream;\n stream.open();\n\n return Promise.resolve(stream);\n }\n\n closeStream(panelId) {\n var stream = this.openStreams[panelId];\n if (stream) {\n stream.close();\n delete this.openStreams[panelId];\n }\n }\n}\n"]} -------------------------------------------------------------------------------- /dist/module.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | Object.defineProperty(exports, "__esModule", { 4 | value: true 5 | }); 6 | exports.ConfigCtrl = exports.QueryCtrl = exports.Datasource = undefined; 7 | 8 | var _datasource = require('./datasource'); 9 | 10 | var _query_ctrl = require('./query_ctrl'); 11 | 12 | function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } 13 | 14 | var StreamingConfigCtrl = function StreamingConfigCtrl() { 15 | _classCallCheck(this, StreamingConfigCtrl); 16 | }; 17 | 18 | StreamingConfigCtrl.templateUrl = 'partials/config.html'; 19 | 20 | exports.Datasource = _datasource.StreamingDatasource; 21 | exports.QueryCtrl = _query_ctrl.StreamingQueryCtrl; 22 | exports.ConfigCtrl = StreamingConfigCtrl; 23 | //# sourceMappingURL=module.js.map 24 | -------------------------------------------------------------------------------- /dist/module.js.map: -------------------------------------------------------------------------------- 1 | {"version":3,"sources":["../src/module.js"],"names":["StreamingConfigCtrl","templateUrl","Datasource","StreamingDatasource","QueryCtrl","StreamingQueryCtrl","ConfigCtrl"],"mappings":";;;;;;;AAAA;;AACA;;;;IAEMA,mB;;;;AACNA,oBAAoBC,WAApB,GAAkC,sBAAlC;;QAEgCC,U,GAAvBC,+B;QAAyDC,S,GAAtBC,8B;QAAwDC,U,GAAvBN,mB","file":"module.js","sourcesContent":["import { StreamingDatasource } from './datasource';\nimport { StreamingQueryCtrl } from './query_ctrl';\n\nclass StreamingConfigCtrl {}\nStreamingConfigCtrl.templateUrl = 'partials/config.html';\n\nexport { StreamingDatasource as Datasource, StreamingQueryCtrl as QueryCtrl, StreamingConfigCtrl as ConfigCtrl };\n"]} -------------------------------------------------------------------------------- /dist/partials/config.html: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /dist/partials/query.editor.html: -------------------------------------------------------------------------------- 1 | 2 |
3 |
4 | 11 |
12 |
13 |
14 |
15 |
16 |
17 | -------------------------------------------------------------------------------- /dist/plugin.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "Simple Streaming Datasource", 3 | "id": "simple-streaming-datasource", 4 | "type": "datasource", 5 | 6 | "metrics": true, 7 | "annotations": false, 8 | 9 | "includes": [{ "type": "dashboard", "name": "Streaming Demo Dashboard", "path": "dashboards/demodashboard.json" }], 10 | 11 | "info": { 12 | "description": "An end to end example of live streaming data in Grafana", 13 | "author": { 14 | "name": "Sean Lafferty", 15 | "url": "https://grafana.com" 16 | }, 17 | "links": [ 18 | { "name": "GitHub", "url": "https://github.com/seanlaff/simple-streaming-datasource" }, 19 | { "name": "MIT License", "url": "https://github.com/seanlaff/simple-streaming-datasource/blob/master/LICENSE" } 20 | ], 21 | "version": "1.0.0", 22 | "updated": "2019-03-01" 23 | }, 24 | 25 | "dependencies": { 26 | "grafanaVersion": "5.x.x" 27 | } 28 | } 29 | -------------------------------------------------------------------------------- /dist/query_ctrl.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | Object.defineProperty(exports, "__esModule", { 4 | value: true 5 | }); 6 | exports.StreamingQueryCtrl = undefined; 7 | 8 | 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; }; }(); 9 | 10 | var _sdk = require('app/plugins/sdk'); 11 | 12 | function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } 13 | 14 | function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; } 15 | 16 | function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } 17 | 18 | var StreamingQueryCtrl = exports.StreamingQueryCtrl = function (_QueryCtrl) { 19 | _inherits(StreamingQueryCtrl, _QueryCtrl); 20 | 21 | function StreamingQueryCtrl($scope, $injector) { 22 | _classCallCheck(this, StreamingQueryCtrl); 23 | 24 | var _this = _possibleConstructorReturn(this, (StreamingQueryCtrl.__proto__ || Object.getPrototypeOf(StreamingQueryCtrl)).call(this, $scope, $injector)); 25 | 26 | _this.scope = $scope; 27 | _this.target.numSeries = _this.target.numSeries || 1; 28 | return _this; 29 | } 30 | 31 | _createClass(StreamingQueryCtrl, [{ 32 | key: 'onChangeInternal', 33 | value: function onChangeInternal() { 34 | this.datasource.closeStream(this.panel.id); 35 | 36 | // So here's the only real hack. We want to kill the old stream and open a new one 37 | // with the new query. Normally, this.panelCtrl.refresh() would take care of shooting 38 | // off a new call to datasource.query() however there's a conditional inside grafana's 39 | // onMetricsPanelRefresh() that prevents calling datasource.query() again if it sees that 40 | // this panel is connected to a datastream. 41 | // 42 | // To get around that, we'll null out the dataStream... which allows onMetricsPanelRefresh() 43 | // to call datasource.query()... which opens a new stream. 44 | // 45 | // Unfortunately this same behavior is why changing the time range will not affect an 46 | // existing streaming panel unless you save and refresh the dashboard. 47 | // 48 | // I'll open a PR to the team and see if I can get some more feedback on why that behavior exists. 49 | this.panelCtrl.dataStream = null; 50 | 51 | this.panelCtrl.refresh(); // Asks the panel to refresh data. 52 | } 53 | }]); 54 | 55 | return StreamingQueryCtrl; 56 | }(_sdk.QueryCtrl); 57 | 58 | StreamingQueryCtrl.templateUrl = 'partials/query.editor.html'; 59 | //# sourceMappingURL=query_ctrl.js.map 60 | -------------------------------------------------------------------------------- /dist/query_ctrl.js.map: -------------------------------------------------------------------------------- 1 | {"version":3,"sources":["../src/query_ctrl.js"],"names":["StreamingQueryCtrl","$scope","$injector","scope","target","numSeries","datasource","closeStream","panel","id","panelCtrl","dataStream","refresh","QueryCtrl","templateUrl"],"mappings":";;;;;;;;;AAAA;;;;;;;;IAEaA,kB,WAAAA,kB;;;AACX,8BAAYC,MAAZ,EAAoBC,SAApB,EAA+B;AAAA;;AAAA,wIACvBD,MADuB,EACfC,SADe;;AAG7B,UAAKC,KAAL,GAAaF,MAAb;AACA,UAAKG,MAAL,CAAYC,SAAZ,GAAwB,MAAKD,MAAL,CAAYC,SAAZ,IAAyB,CAAjD;AAJ6B;AAK9B;;;;uCAEkB;AACjB,WAAKC,UAAL,CAAgBC,WAAhB,CAA4B,KAAKC,KAAL,CAAWC,EAAvC;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAKC,SAAL,CAAeC,UAAf,GAA4B,IAA5B;;AAEA,WAAKD,SAAL,CAAeE,OAAf,GAlBiB,CAkBS;AAC3B;;;;EA3BqCC,c;;AA8BxCb,mBAAmBc,WAAnB,GAAiC,4BAAjC","file":"query_ctrl.js","sourcesContent":["import { QueryCtrl } from 'app/plugins/sdk';\n\nexport class StreamingQueryCtrl extends QueryCtrl {\n constructor($scope, $injector) {\n super($scope, $injector);\n\n this.scope = $scope;\n this.target.numSeries = this.target.numSeries || 1;\n }\n\n onChangeInternal() {\n this.datasource.closeStream(this.panel.id);\n\n // So here's the only real hack. We want to kill the old stream and open a new one\n // with the new query. Normally, this.panelCtrl.refresh() would take care of shooting\n // off a new call to datasource.query() however there's a conditional inside grafana's\n // onMetricsPanelRefresh() that prevents calling datasource.query() again if it sees that\n // this panel is connected to a datastream.\n //\n // To get around that, we'll null out the dataStream... which allows onMetricsPanelRefresh()\n // to call datasource.query()... which opens a new stream.\n //\n // Unfortunately this same behavior is why changing the time range will not affect an\n // existing streaming panel unless you save and refresh the dashboard.\n //\n // I'll open a PR to the team and see if I can get some more feedback on why that behavior exists.\n this.panelCtrl.dataStream = null;\n\n this.panelCtrl.refresh(); // Asks the panel to refresh data.\n }\n}\n\nStreamingQueryCtrl.templateUrl = 'partials/query.editor.html';\n"]} -------------------------------------------------------------------------------- /dist/stream_handler.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | Object.defineProperty(exports, "__esModule", { 4 | value: true 5 | }); 6 | exports.StreamHandler = undefined; 7 | 8 | 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; }; }(); 9 | 10 | var _moment = require('moment'); 11 | 12 | var _moment2 = _interopRequireDefault(_moment); 13 | 14 | var _ndjson = require('./vendor/ndjson.js'); 15 | 16 | var ndjsonStream = _interopRequireWildcard(_ndjson); 17 | 18 | var _rxjsUmdMin = require('./vendor/rxjs.umd.min.js'); 19 | 20 | var rxjs = _interopRequireWildcard(_rxjsUmdMin); 21 | 22 | function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } else { var newObj = {}; if (obj != null) { for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) newObj[key] = obj[key]; } } newObj.default = obj; return newObj; } } 23 | 24 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } 25 | 26 | function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } } 27 | 28 | function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } 29 | 30 | // We return a StreamHandler wrapped in a promise from the datasource's 31 | // Query method. Grafana expects this object to have a `subscribe` method, 32 | // which it reads live data from. 33 | var StreamHandler = exports.StreamHandler = function () { 34 | function StreamHandler(options, datasource) { 35 | var _this = this; 36 | 37 | _classCallCheck(this, StreamHandler); 38 | 39 | this.options = options; 40 | this.ds = datasource; 41 | this.subject = new rxjs.Subject(); // Where we'll publish our data 42 | this.subscribe = function (options) { 43 | // To avoid destroying the browser with repaints, add a throttle (You may want to tweak this) 44 | var throttledSubject = _this.subject.pipe(rxjs.operators.throttleTime(100)); 45 | return throttledSubject.subscribe(options); 46 | }; 47 | this.reader = null; 48 | this.metrics = {}; // A local copy of our data that we'll operate on before sending to the rxjs Subject 49 | } 50 | 51 | _createClass(StreamHandler, [{ 52 | key: 'open', 53 | value: function open() { 54 | var _this2 = this; 55 | 56 | var request = new Request(this.ds.url + '?numSeries=' + this.options.targets[0].numSeries); 57 | fetch(request).then(function (response) { 58 | // In the real world its likely that our json gets chopped into 59 | // chunks when streamed from the backend. ndjsonStream handles 60 | // reconstructing the newline-delimmited json for us. 61 | return ndjsonStream.default(response.body); 62 | }).then(function (s) { 63 | _this2.reader = s.getReader(); // Save the reader so we can cancel it later 64 | var _readHandler = void 0; 65 | 66 | _this2.reader.read().then(_readHandler = function readHandler(result) { 67 | if (result.done) { 68 | return; 69 | } 70 | _this2.handleMessage(result.value); 71 | _this2.reader.read().then(_readHandler); 72 | }); 73 | }); 74 | } 75 | }, { 76 | key: 'handleMessage', 77 | value: function handleMessage(msg) { 78 | var _this3 = this; 79 | 80 | var oldestTimestamp = this.options.range.from.unix() * 1000; 81 | var mostRecentTimestamp = this.options.range.to.unix() * 1000; 82 | 83 | // Assuming the data we're being streamed in chronologically ordered 84 | if (msg.timestamp > mostRecentTimestamp) { 85 | mostRecentTimestamp = msg.timestamp; 86 | } 87 | 88 | // See if we have any data already for this target 89 | var series = this.metrics[msg.series]; 90 | if (!series) { 91 | series = { target: msg.series, datapoints: [] }; 92 | this.metrics[msg.series] = series; 93 | } 94 | series.datapoints = [].concat(_toConsumableArray(series.datapoints), [[msg.value, msg.timestamp]]); // Add our new point to the end 95 | 96 | // Slide the "window" by removing any points that are older than the latest point, 97 | // minus the width of the current time range 98 | series.datapoints = series.datapoints.filter(function (p) { 99 | return p[1] > mostRecentTimestamp - (_this3.options.range.to.unix() * 1000 - _this3.options.range.from.unix() * 1000); 100 | }); 101 | 102 | // Grab the timestamp of the earliest point still in the datapoints array, we'll 103 | // move the time window forward to match it 104 | if (series.datapoints[0] && series.datapoints[0][1] && series.datapoints[0][1] > oldestTimestamp) { 105 | oldestTimestamp = series.datapoints[0][1]; 106 | } 107 | 108 | var ts = Object.keys(this.metrics).map(function (key) { 109 | return _this3.metrics[key]; 110 | }); 111 | 112 | this.subject.next({ 113 | data: ts, 114 | range: { from: (0, _moment2.default)(oldestTimestamp), to: (0, _moment2.default)(mostRecentTimestamp) } 115 | }); 116 | } 117 | }, { 118 | key: 'close', 119 | value: function close() { 120 | if (this.reader) { 121 | this.reader.cancel('Close was called on streamHandler'); 122 | } 123 | } 124 | }]); 125 | 126 | return StreamHandler; 127 | }(); 128 | //# sourceMappingURL=stream_handler.js.map 129 | -------------------------------------------------------------------------------- /dist/stream_handler.js.map: -------------------------------------------------------------------------------- 1 | {"version":3,"sources":["../src/stream_handler.js"],"names":["ndjsonStream","rxjs","StreamHandler","options","datasource","ds","subject","Subject","subscribe","throttledSubject","pipe","operators","throttleTime","reader","metrics","request","Request","url","targets","numSeries","fetch","then","default","response","body","s","getReader","readHandler","read","result","done","handleMessage","value","msg","oldestTimestamp","range","from","unix","mostRecentTimestamp","to","timestamp","series","target","datapoints","filter","p","ts","Object","keys","map","key","next","data","cancel"],"mappings":";;;;;;;;;AAAA;;;;AACA;;IAAYA,Y;;AACZ;;IAAYC,I;;;;;;;;;;AAEZ;AACA;AACA;IACaC,a,WAAAA,a;AACX,yBAAYC,OAAZ,EAAqBC,UAArB,EAAiC;AAAA;;AAAA;;AAC/B,SAAKD,OAAL,GAAeA,OAAf;AACA,SAAKE,EAAL,GAAUD,UAAV;AACA,SAAKE,OAAL,GAAe,IAAIL,KAAKM,OAAT,EAAf,CAH+B,CAGI;AACnC,SAAKC,SAAL,GAAiB,mBAAW;AAC1B;AACA,UAAIC,mBAAmB,MAAKH,OAAL,CAAaI,IAAb,CAAkBT,KAAKU,SAAL,CAAeC,YAAf,CAA4B,GAA5B,CAAlB,CAAvB;AACA,aAAOH,iBAAiBD,SAAjB,CAA2BL,OAA3B,CAAP;AACD,KAJD;AAKA,SAAKU,MAAL,GAAc,IAAd;AACA,SAAKC,OAAL,GAAe,EAAf,CAV+B,CAUZ;AACpB;;;;2BAEM;AAAA;;AACL,UAAIC,UAAU,IAAIC,OAAJ,CAAe,KAAKX,EAAL,CAAQY,GAAvB,mBAAwC,KAAKd,OAAL,CAAae,OAAb,CAAqB,CAArB,EAAwBC,SAAhE,CAAd;AACAC,YAAML,OAAN,EACGM,IADH,CACQ,oBAAY;AAChB;AACA;AACA;AACA,eAAOrB,aAAasB,OAAb,CAAqBC,SAASC,IAA9B,CAAP;AACD,OANH,EAOGH,IAPH,CAOQ,aAAK;AACT,eAAKR,MAAL,GAAcY,EAAEC,SAAF,EAAd,CADS,CACoB;AAC7B,YAAIC,qBAAJ;;AAEA,eAAKd,MAAL,CAAYe,IAAZ,GAAmBP,IAAnB,CACGM,eAAc,6BAAU;AACvB,cAAIE,OAAOC,IAAX,EAAiB;AACf;AACD;AACD,iBAAKC,aAAL,CAAmBF,OAAOG,KAA1B;AACA,iBAAKnB,MAAL,CAAYe,IAAZ,GAAmBP,IAAnB,CAAwBM,YAAxB;AACD,SAPH;AASD,OApBH;AAqBD;;;kCAEaM,G,EAAK;AAAA;;AACjB,UAAIC,kBAAkB,KAAK/B,OAAL,CAAagC,KAAb,CAAmBC,IAAnB,CAAwBC,IAAxB,KAAiC,IAAvD;AACA,UAAIC,sBAAsB,KAAKnC,OAAL,CAAagC,KAAb,CAAmBI,EAAnB,CAAsBF,IAAtB,KAA+B,IAAzD;;AAEA;AACA,UAAIJ,IAAIO,SAAJ,GAAgBF,mBAApB,EAAyC;AACvCA,8BAAsBL,IAAIO,SAA1B;AACD;;AAED;AACA,UAAIC,SAAS,KAAK3B,OAAL,CAAamB,IAAIQ,MAAjB,CAAb;AACA,UAAI,CAACA,MAAL,EAAa;AACXA,iBAAS,EAAEC,QAAQT,IAAIQ,MAAd,EAAsBE,YAAY,EAAlC,EAAT;AACA,aAAK7B,OAAL,CAAamB,IAAIQ,MAAjB,IAA2BA,MAA3B;AACD;AACDA,aAAOE,UAAP,gCAAwBF,OAAOE,UAA/B,IAA2C,CAACV,IAAID,KAAL,EAAYC,IAAIO,SAAhB,CAA3C,GAfiB,CAeuD;;AAExE;AACA;AACAC,aAAOE,UAAP,GAAoBF,OAAOE,UAAP,CAAkBC,MAAlB,CAClB;AAAA,eAAKC,EAAE,CAAF,IAAOP,uBAAuB,OAAKnC,OAAL,CAAagC,KAAb,CAAmBI,EAAnB,CAAsBF,IAAtB,KAA+B,IAA/B,GAAsC,OAAKlC,OAAL,CAAagC,KAAb,CAAmBC,IAAnB,CAAwBC,IAAxB,KAAiC,IAA9F,CAAZ;AAAA,OADkB,CAApB;;AAIA;AACA;AACA,UAAII,OAAOE,UAAP,CAAkB,CAAlB,KAAwBF,OAAOE,UAAP,CAAkB,CAAlB,EAAqB,CAArB,CAAxB,IAAmDF,OAAOE,UAAP,CAAkB,CAAlB,EAAqB,CAArB,IAA0BT,eAAjF,EAAkG;AAChGA,0BAAkBO,OAAOE,UAAP,CAAkB,CAAlB,EAAqB,CAArB,CAAlB;AACD;;AAED,UAAMG,KAAKC,OAAOC,IAAP,CAAY,KAAKlC,OAAjB,EAA0BmC,GAA1B,CAA8B,eAAO;AAC9C,eAAO,OAAKnC,OAAL,CAAaoC,GAAb,CAAP;AACD,OAFU,CAAX;;AAIA,WAAK5C,OAAL,CAAa6C,IAAb,CAAkB;AAChBC,cAAMN,EADU;AAEhBX,eAAO,EAAEC,MAAM,sBAAOF,eAAP,CAAR,EAAiCK,IAAI,sBAAOD,mBAAP,CAArC;AAFS,OAAlB;AAID;;;4BAEO;AACN,UAAI,KAAKzB,MAAT,EAAiB;AACf,aAAKA,MAAL,CAAYwC,MAAZ,CAAmB,mCAAnB;AACD;AACF","file":"stream_handler.js","sourcesContent":["import moment from 'moment';\nimport * as ndjsonStream from './vendor/ndjson.js';\nimport * as rxjs from './vendor/rxjs.umd.min.js';\n\n// We return a StreamHandler wrapped in a promise from the datasource's\n// Query method. Grafana expects this object to have a `subscribe` method,\n// which it reads live data from.\nexport class StreamHandler {\n constructor(options, datasource) {\n this.options = options;\n this.ds = datasource;\n this.subject = new rxjs.Subject(); // Where we'll publish our data\n this.subscribe = options => {\n // To avoid destroying the browser with repaints, add a throttle (You may want to tweak this)\n var throttledSubject = this.subject.pipe(rxjs.operators.throttleTime(100));\n return throttledSubject.subscribe(options);\n };\n this.reader = null;\n this.metrics = {}; // A local copy of our data that we'll operate on before sending to the rxjs Subject\n }\n\n open() {\n var request = new Request(`${this.ds.url}?numSeries=${this.options.targets[0].numSeries}`);\n fetch(request)\n .then(response => {\n // In the real world its likely that our json gets chopped into\n // chunks when streamed from the backend. ndjsonStream handles\n // reconstructing the newline-delimmited json for us.\n return ndjsonStream.default(response.body);\n })\n .then(s => {\n this.reader = s.getReader(); // Save the reader so we can cancel it later\n let readHandler;\n\n this.reader.read().then(\n (readHandler = result => {\n if (result.done) {\n return;\n }\n this.handleMessage(result.value);\n this.reader.read().then(readHandler);\n })\n );\n });\n }\n\n handleMessage(msg) {\n let oldestTimestamp = this.options.range.from.unix() * 1000;\n var mostRecentTimestamp = this.options.range.to.unix() * 1000;\n\n // Assuming the data we're being streamed in chronologically ordered\n if (msg.timestamp > mostRecentTimestamp) {\n mostRecentTimestamp = msg.timestamp;\n }\n\n // See if we have any data already for this target\n let series = this.metrics[msg.series];\n if (!series) {\n series = { target: msg.series, datapoints: [] };\n this.metrics[msg.series] = series;\n }\n series.datapoints = [...series.datapoints, [msg.value, msg.timestamp]]; // Add our new point to the end\n\n // Slide the \"window\" by removing any points that are older than the latest point,\n // minus the width of the current time range\n series.datapoints = series.datapoints.filter(\n p => p[1] > mostRecentTimestamp - (this.options.range.to.unix() * 1000 - this.options.range.from.unix() * 1000)\n );\n\n // Grab the timestamp of the earliest point still in the datapoints array, we'll\n // move the time window forward to match it\n if (series.datapoints[0] && series.datapoints[0][1] && series.datapoints[0][1] > oldestTimestamp) {\n oldestTimestamp = series.datapoints[0][1];\n }\n\n const ts = Object.keys(this.metrics).map(key => {\n return this.metrics[key];\n });\n\n this.subject.next({\n data: ts,\n range: { from: moment(oldestTimestamp), to: moment(mostRecentTimestamp) },\n });\n }\n\n close() {\n if (this.reader) {\n this.reader.cancel('Close was called on streamHandler');\n }\n }\n}\n"]} -------------------------------------------------------------------------------- /dist/vendor/ndjson.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | /*exported ndjsonStream*/ 3 | 4 | var namespace = (window.canNamespace = {}); 5 | 6 | var ndjsonStream = function(response) { 7 | // For cancellation 8 | var is_reader, 9 | cancellationRequest = false; 10 | return new ReadableStream({ 11 | start: function(controller) { 12 | var reader = response.getReader(); 13 | is_reader = reader; 14 | var decoder = new TextDecoder(); 15 | var data_buf = ''; 16 | 17 | reader.read().then(function processResult(result) { 18 | if (result.done) { 19 | if (cancellationRequest) { 20 | // Immediately exit 21 | return; 22 | } 23 | 24 | data_buf = data_buf.trim(); 25 | if (data_buf.length !== 0) { 26 | try { 27 | var data_l = JSON.parse(data_buf); 28 | controller.enqueue(data_l); 29 | } catch (e) { 30 | controller.error(e); 31 | return; 32 | } 33 | } 34 | controller.close(); 35 | return; 36 | } 37 | 38 | var data = decoder.decode(result.value, { stream: true }); 39 | data_buf += data; 40 | var lines = data_buf.split('\n'); 41 | for (var i = 0; i < lines.length - 1; ++i) { 42 | var l = lines[i].trim(); 43 | if (l.length > 0) { 44 | try { 45 | var data_line = JSON.parse(l); 46 | controller.enqueue(data_line); 47 | } catch (e) { 48 | controller.error(e); 49 | cancellationRequest = true; 50 | reader.cancel(); 51 | return; 52 | } 53 | } 54 | } 55 | data_buf = lines[lines.length - 1]; 56 | 57 | return reader.read().then(processResult); 58 | }); 59 | }, 60 | cancel: function(reason) { 61 | console.log('Cancel registered due to ', reason); 62 | cancellationRequest = true; 63 | is_reader.cancel(); 64 | }, 65 | }); 66 | }; 67 | 68 | module.exports = namespace.ndjsonStream = ndjsonStream; 69 | -------------------------------------------------------------------------------- /livedata.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/seanlaff/simple-streaming-datasource/cc5115246f5b5e4f56c12887dffab3bb4fc62432/livedata.gif -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "simple-streaming-datasource", 3 | "private": false, 4 | "version": "1.0.0", 5 | "description": "An end to end example of live streaming data in Grafana", 6 | "main": "index.js", 7 | "scripts": { 8 | "build": "./node_modules/grunt-cli/bin/grunt" 9 | }, 10 | "repository": { 11 | "type": "git", 12 | "url": "git+https://github.com/seanlaff/simple-streaming-datasource.git" 13 | }, 14 | "author": "Sean Lafferty", 15 | "license": "MIT", 16 | "bugs": { 17 | "url": "https://github.com/seanlaff/simple-streaming-datasource/issues" 18 | }, 19 | "engineStrict": true, 20 | "devDependencies": { 21 | "babel": "^6.23.0", 22 | "babel-plugin-transform-object-rest-spread": "^6.26.0", 23 | "babel-preset-env": "^1.7.0", 24 | "grunt": "^1.0.1", 25 | "grunt-babel": "~6.0.0", 26 | "grunt-cli": "^1.2.0", 27 | "grunt-contrib-clean": "^1.1.0", 28 | "grunt-contrib-copy": "^1.0.0", 29 | "grunt-contrib-uglify": "^2.3.0", 30 | "grunt-contrib-watch": "^1.0.0", 31 | "grunt-execute": "~0.2.2", 32 | "grunt-systemjs-builder": "^1.0.0", 33 | "load-grunt-tasks": "^3.5.2" 34 | }, 35 | "dependencies": { 36 | "moment": "^2.24.0" 37 | }, 38 | "homepage": "https://github.com/seanlaff/simple-streaming-datasource#readme" 39 | } 40 | -------------------------------------------------------------------------------- /server.go: -------------------------------------------------------------------------------- 1 | package main 2 | 3 | import ( 4 | "encoding/json" 5 | "fmt" 6 | "math/rand" 7 | "net/http" 8 | "strconv" 9 | "time" 10 | ) 11 | 12 | type datapoint struct { 13 | Series int `json:"series"` 14 | Timestamp int64 `json:"timestamp"` 15 | Value int `json:"value"` 16 | } 17 | 18 | func main() { 19 | http.HandleFunc("/", handler) 20 | if err := http.ListenAndServe(":8080", nil); err != nil { 21 | panic(err) 22 | } 23 | 24 | } 25 | 26 | func handler(w http.ResponseWriter, r *http.Request) { 27 | flusher, ok := w.(http.Flusher) 28 | if !ok { 29 | panic("expected http.ResponseWriter to be an http.Flusher") 30 | } 31 | 32 | // How many different metrics to generate 33 | numSeries := 1 34 | numSeriesParam := r.URL.Query().Get("numSeries") 35 | if numSeriesParam != "" { 36 | numSeries, _ = strconv.Atoi(numSeriesParam) 37 | } 38 | 39 | ticker := time.NewTicker(50 * time.Millisecond) 40 | for { 41 | select { 42 | case t := <-ticker.C: 43 | for i := 0; i < numSeries; i++ { 44 | currentPoint := &datapoint{ 45 | Series: i, 46 | Timestamp: t.UnixNano() / 1000000, // JS likes ms timestamps 47 | Value: rand.Intn(10) + 10*i, 48 | } 49 | j, _ := json.Marshal(currentPoint) 50 | fmt.Fprintf(w, "%s\n", j) 51 | flusher.Flush() // Trigger "chunked" encoding and send a chunk... 52 | } 53 | } 54 | } 55 | 56 | } 57 | -------------------------------------------------------------------------------- /src/dashboards/demodashboard.json: -------------------------------------------------------------------------------- 1 | { 2 | "annotations": { 3 | "list": [ 4 | { 5 | "builtIn": 1, 6 | "datasource": "-- Grafana --", 7 | "enable": true, 8 | "hide": true, 9 | "iconColor": "rgba(0, 211, 255, 1)", 10 | "name": "Annotations & Alerts", 11 | "type": "dashboard" 12 | } 13 | ] 14 | }, 15 | "editable": true, 16 | "gnetId": null, 17 | "graphTooltip": 0, 18 | "id": 2, 19 | "links": [], 20 | "panels": [ 21 | { 22 | "cacheTimeout": null, 23 | "colorBackground": false, 24 | "colorValue": false, 25 | "colors": ["#299c46", "rgba(237, 129, 40, 0.89)", "#d44a3a"], 26 | "datasource": "Simple Streaming Datasource", 27 | "decimals": 2, 28 | "format": "none", 29 | "gauge": { 30 | "maxValue": 10, 31 | "minValue": 0, 32 | "show": true, 33 | "thresholdLabels": false, 34 | "thresholdMarkers": true 35 | }, 36 | "gridPos": { 37 | "h": 7, 38 | "w": 8, 39 | "x": 0, 40 | "y": 0 41 | }, 42 | "id": 4, 43 | "interval": null, 44 | "links": [], 45 | "mappingType": 1, 46 | "mappingTypes": [ 47 | { 48 | "name": "value to text", 49 | "value": 1 50 | }, 51 | { 52 | "name": "range to text", 53 | "value": 2 54 | } 55 | ], 56 | "maxDataPoints": 100, 57 | "nullPointMode": "connected", 58 | "nullText": null, 59 | "postfix": "", 60 | "postfixFontSize": "50%", 61 | "prefix": "", 62 | "prefixFontSize": "50%", 63 | "rangeMaps": [ 64 | { 65 | "from": "null", 66 | "text": "N/A", 67 | "to": "null" 68 | } 69 | ], 70 | "sparkline": { 71 | "fillColor": "rgba(31, 118, 189, 0.18)", 72 | "full": false, 73 | "lineColor": "rgb(31, 120, 193)", 74 | "show": false 75 | }, 76 | "tableColumn": "", 77 | "targets": [ 78 | { 79 | "numSeries": 1, 80 | "refId": "A" 81 | } 82 | ], 83 | "thresholds": "", 84 | "title": "Panel Title", 85 | "type": "singlestat", 86 | "valueFontSize": "80%", 87 | "valueMaps": [ 88 | { 89 | "op": "=", 90 | "text": "N/A", 91 | "value": "null" 92 | } 93 | ], 94 | "valueName": "avg" 95 | }, 96 | { 97 | "aliasColors": {}, 98 | "bars": false, 99 | "dashLength": 10, 100 | "dashes": false, 101 | "datasource": "Simple Streaming Datasource", 102 | "fill": 1, 103 | "gridPos": { 104 | "h": 7, 105 | "w": 16, 106 | "x": 8, 107 | "y": 0 108 | }, 109 | "id": 2, 110 | "legend": { 111 | "avg": false, 112 | "current": false, 113 | "max": false, 114 | "min": false, 115 | "show": true, 116 | "total": false, 117 | "values": false 118 | }, 119 | "lines": true, 120 | "linewidth": 1, 121 | "nullPointMode": "null", 122 | "percentage": false, 123 | "pointradius": 5, 124 | "points": false, 125 | "renderer": "flot", 126 | "seriesOverrides": [], 127 | "spaceLength": 10, 128 | "stack": false, 129 | "steppedLine": false, 130 | "targets": [ 131 | { 132 | "numSeries": 4, 133 | "refId": "A", 134 | "target": "select metrich", 135 | "type": "timeserie" 136 | } 137 | ], 138 | "thresholds": [], 139 | "timeFrom": null, 140 | "timeRegions": [], 141 | "timeShift": null, 142 | "title": "Panel Title", 143 | "tooltip": { 144 | "shared": true, 145 | "sort": 0, 146 | "value_type": "individual" 147 | }, 148 | "type": "graph", 149 | "xaxis": { 150 | "buckets": null, 151 | "mode": "time", 152 | "name": null, 153 | "show": true, 154 | "values": [] 155 | }, 156 | "yaxes": [ 157 | { 158 | "format": "short", 159 | "label": null, 160 | "logBase": 1, 161 | "max": null, 162 | "min": null, 163 | "show": true 164 | }, 165 | { 166 | "format": "short", 167 | "label": null, 168 | "logBase": 1, 169 | "max": null, 170 | "min": null, 171 | "show": true 172 | } 173 | ], 174 | "yaxis": { 175 | "align": false, 176 | "alignLevel": null 177 | } 178 | } 179 | ], 180 | "schemaVersion": 16, 181 | "style": "dark", 182 | "tags": [], 183 | "templating": { 184 | "list": [] 185 | }, 186 | "time": { 187 | "from": "now-30s", 188 | "to": "now" 189 | }, 190 | "timepicker": { 191 | "refresh_intervals": ["5s", "10s", "30s", "1m", "5m", "15m", "30m", "1h", "2h", "1d"], 192 | "time_options": ["5m", "15m", "1h", "6h", "12h", "24h", "2d", "7d", "30d"] 193 | }, 194 | "timezone": "", 195 | "title": "Streaming Demo Dashboard", 196 | "uid": "kfYJBgCmk", 197 | "version": 2 198 | } 199 | -------------------------------------------------------------------------------- /src/datasource.js: -------------------------------------------------------------------------------- 1 | import { StreamHandler } from './stream_handler'; 2 | 3 | export class StreamingDatasource { 4 | constructor(instanceSettings, $q) { 5 | this.url = instanceSettings.url; 6 | this.q = $q; 7 | this.headers = { 'Content-Type': 'application/json' }; 8 | 9 | // Keep a list of open streams so we can cancel them when querys/timerange change 10 | this.openStreams = []; 11 | } 12 | 13 | query(options) { 14 | var stream = this.openStreams[options.panelId]; 15 | if (stream) { 16 | return Promise.resolve(stream); 17 | } 18 | 19 | stream = new StreamHandler(options, this); 20 | this.openStreams[options.panelId] = stream; 21 | stream.open(); 22 | 23 | return Promise.resolve(stream); 24 | } 25 | 26 | closeStream(panelId) { 27 | var stream = this.openStreams[panelId]; 28 | if (stream) { 29 | stream.close(); 30 | delete this.openStreams[panelId]; 31 | } 32 | } 33 | } 34 | -------------------------------------------------------------------------------- /src/module.js: -------------------------------------------------------------------------------- 1 | import { StreamingDatasource } from './datasource'; 2 | import { StreamingQueryCtrl } from './query_ctrl'; 3 | 4 | class StreamingConfigCtrl {} 5 | StreamingConfigCtrl.templateUrl = 'partials/config.html'; 6 | 7 | export { StreamingDatasource as Datasource, StreamingQueryCtrl as QueryCtrl, StreamingConfigCtrl as ConfigCtrl }; 8 | -------------------------------------------------------------------------------- /src/partials/config.html: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /src/partials/query.editor.html: -------------------------------------------------------------------------------- 1 | 2 |
3 |
4 | 11 |
12 |
13 |
14 |
15 |
16 |
17 | -------------------------------------------------------------------------------- /src/plugin.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "Simple Streaming Datasource", 3 | "id": "simple-streaming-datasource", 4 | "type": "datasource", 5 | 6 | "metrics": true, 7 | "annotations": false, 8 | 9 | "includes": [{ "type": "dashboard", "name": "Streaming Demo Dashboard", "path": "dashboards/demodashboard.json" }], 10 | 11 | "info": { 12 | "description": "An end to end example of live streaming data in Grafana", 13 | "author": { 14 | "name": "Sean Lafferty", 15 | "url": "https://grafana.com" 16 | }, 17 | "links": [ 18 | { "name": "GitHub", "url": "https://github.com/seanlaff/simple-streaming-datasource" }, 19 | { "name": "MIT License", "url": "https://github.com/seanlaff/simple-streaming-datasource/blob/master/LICENSE" } 20 | ], 21 | "version": "1.0.0", 22 | "updated": "2019-03-01" 23 | }, 24 | 25 | "dependencies": { 26 | "grafanaVersion": "5.x.x" 27 | } 28 | } 29 | -------------------------------------------------------------------------------- /src/query_ctrl.js: -------------------------------------------------------------------------------- 1 | import { QueryCtrl } from 'app/plugins/sdk'; 2 | 3 | export class StreamingQueryCtrl extends QueryCtrl { 4 | constructor($scope, $injector) { 5 | super($scope, $injector); 6 | 7 | this.scope = $scope; 8 | this.target.numSeries = this.target.numSeries || 1; 9 | } 10 | 11 | onChangeInternal() { 12 | this.datasource.closeStream(this.panel.id); 13 | 14 | // So here's the only real hack. We want to kill the old stream and open a new one 15 | // with the new query. Normally, this.panelCtrl.refresh() would take care of shooting 16 | // off a new call to datasource.query() however there's a conditional inside grafana's 17 | // onMetricsPanelRefresh() that prevents calling datasource.query() again if it sees that 18 | // this panel is connected to a datastream. 19 | // 20 | // To get around that, we'll null out the dataStream... which allows onMetricsPanelRefresh() 21 | // to call datasource.query()... which opens a new stream. 22 | // 23 | // Unfortunately this same behavior is why changing the time range will not affect an 24 | // existing streaming panel unless you save and refresh the dashboard. 25 | // 26 | // I'll open a PR to the team and see if I can get some more feedback on why that behavior exists. 27 | this.panelCtrl.dataStream = null; 28 | 29 | this.panelCtrl.refresh(); // Asks the panel to refresh data. 30 | } 31 | } 32 | 33 | StreamingQueryCtrl.templateUrl = 'partials/query.editor.html'; 34 | -------------------------------------------------------------------------------- /src/stream_handler.js: -------------------------------------------------------------------------------- 1 | import moment from 'moment'; 2 | import * as ndjsonStream from './vendor/ndjson.js'; 3 | import * as rxjs from './vendor/rxjs.umd.min.js'; 4 | 5 | // We return a StreamHandler wrapped in a promise from the datasource's 6 | // Query method. Grafana expects this object to have a `subscribe` method, 7 | // which it reads live data from. 8 | export class StreamHandler { 9 | constructor(options, datasource) { 10 | this.options = options; 11 | this.ds = datasource; 12 | this.subject = new rxjs.Subject(); // Where we'll publish our data 13 | this.subscribe = options => { 14 | // To avoid destroying the browser with repaints, add a throttle (You may want to tweak this) 15 | var throttledSubject = this.subject.pipe(rxjs.operators.throttleTime(100)); 16 | return throttledSubject.subscribe(options); 17 | }; 18 | this.reader = null; 19 | this.metrics = {}; // A local copy of our data that we'll operate on before sending to the rxjs Subject 20 | } 21 | 22 | open() { 23 | var request = new Request(`${this.ds.url}?numSeries=${this.options.targets[0].numSeries}`); 24 | fetch(request) 25 | .then(response => { 26 | // In the real world its likely that our json gets chopped into 27 | // chunks when streamed from the backend. ndjsonStream handles 28 | // reconstructing the newline-delimmited json for us. 29 | return ndjsonStream.default(response.body); 30 | }) 31 | .then(s => { 32 | this.reader = s.getReader(); // Save the reader so we can cancel it later 33 | let readHandler; 34 | 35 | this.reader.read().then( 36 | (readHandler = result => { 37 | if (result.done) { 38 | return; 39 | } 40 | this.handleMessage(result.value); 41 | this.reader.read().then(readHandler); 42 | }) 43 | ); 44 | }); 45 | } 46 | 47 | handleMessage(msg) { 48 | let oldestTimestamp = this.options.range.from.unix() * 1000; 49 | var mostRecentTimestamp = this.options.range.to.unix() * 1000; 50 | 51 | // Assuming the data we're being streamed in chronologically ordered 52 | if (msg.timestamp > mostRecentTimestamp) { 53 | mostRecentTimestamp = msg.timestamp; 54 | } 55 | 56 | // See if we have any data already for this target 57 | let series = this.metrics[msg.series]; 58 | if (!series) { 59 | series = { target: msg.series, datapoints: [] }; 60 | this.metrics[msg.series] = series; 61 | } 62 | series.datapoints = [...series.datapoints, [msg.value, msg.timestamp]]; // Add our new point to the end 63 | 64 | // Slide the "window" by removing any points that are older than the latest point, 65 | // minus the width of the current time range 66 | series.datapoints = series.datapoints.filter( 67 | p => p[1] > mostRecentTimestamp - (this.options.range.to.unix() * 1000 - this.options.range.from.unix() * 1000) 68 | ); 69 | 70 | // Grab the timestamp of the earliest point still in the datapoints array, we'll 71 | // move the time window forward to match it 72 | if (series.datapoints[0] && series.datapoints[0][1] && series.datapoints[0][1] > oldestTimestamp) { 73 | oldestTimestamp = series.datapoints[0][1]; 74 | } 75 | 76 | const ts = Object.keys(this.metrics).map(key => { 77 | return this.metrics[key]; 78 | }); 79 | 80 | this.subject.next({ 81 | data: ts, 82 | range: { from: moment(oldestTimestamp), to: moment(mostRecentTimestamp) }, 83 | }); 84 | } 85 | 86 | close() { 87 | if (this.reader) { 88 | this.reader.cancel('Close was called on streamHandler'); 89 | } 90 | } 91 | } 92 | -------------------------------------------------------------------------------- /src/vendor/ndjson.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | /*exported ndjsonStream*/ 3 | 4 | var namespace = (window.canNamespace = {}); 5 | 6 | var ndjsonStream = function(response) { 7 | // For cancellation 8 | var is_reader, 9 | cancellationRequest = false; 10 | return new ReadableStream({ 11 | start: function(controller) { 12 | var reader = response.getReader(); 13 | is_reader = reader; 14 | var decoder = new TextDecoder(); 15 | var data_buf = ''; 16 | 17 | reader.read().then(function processResult(result) { 18 | if (result.done) { 19 | if (cancellationRequest) { 20 | // Immediately exit 21 | return; 22 | } 23 | 24 | data_buf = data_buf.trim(); 25 | if (data_buf.length !== 0) { 26 | try { 27 | var data_l = JSON.parse(data_buf); 28 | controller.enqueue(data_l); 29 | } catch (e) { 30 | controller.error(e); 31 | return; 32 | } 33 | } 34 | controller.close(); 35 | return; 36 | } 37 | 38 | var data = decoder.decode(result.value, { stream: true }); 39 | data_buf += data; 40 | var lines = data_buf.split('\n'); 41 | for (var i = 0; i < lines.length - 1; ++i) { 42 | var l = lines[i].trim(); 43 | if (l.length > 0) { 44 | try { 45 | var data_line = JSON.parse(l); 46 | controller.enqueue(data_line); 47 | } catch (e) { 48 | controller.error(e); 49 | cancellationRequest = true; 50 | reader.cancel(); 51 | return; 52 | } 53 | } 54 | } 55 | data_buf = lines[lines.length - 1]; 56 | 57 | return reader.read().then(processResult); 58 | }); 59 | }, 60 | cancel: function(reason) { 61 | console.log('Cancel registered due to ', reason); 62 | cancellationRequest = true; 63 | is_reader.cancel(); 64 | }, 65 | }); 66 | }; 67 | 68 | module.exports = namespace.ndjsonStream = ndjsonStream; 69 | -------------------------------------------------------------------------------- /yarn.lock: -------------------------------------------------------------------------------- 1 | # THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. 2 | # yarn lockfile v1 3 | 4 | 5 | "@types/estree@0.0.38": 6 | version "0.0.38" 7 | resolved "https://registry.yarnpkg.com/@types/estree/-/estree-0.0.38.tgz#c1be40aa933723c608820a99a373a16d215a1ca2" 8 | 9 | "@types/node@*": 10 | version "10.3.6" 11 | resolved "https://registry.yarnpkg.com/@types/node/-/node-10.3.6.tgz#ea8aab9439b59f40d19ec5f13b44642344872b11" 12 | 13 | abbrev@1: 14 | version "1.1.0" 15 | resolved "https://registry.yarnpkg.com/abbrev/-/abbrev-1.1.0.tgz#d0554c2256636e2f56e7c2e5ad183f859428d81f" 16 | 17 | align-text@^0.1.1, align-text@^0.1.3: 18 | version "0.1.4" 19 | resolved "https://registry.yarnpkg.com/align-text/-/align-text-0.1.4.tgz#0cd90a561093f35d0a99256c22b7069433fad117" 20 | dependencies: 21 | kind-of "^3.0.2" 22 | longest "^1.0.1" 23 | repeat-string "^1.5.2" 24 | 25 | amdefine@>=0.0.4: 26 | version "1.0.1" 27 | resolved "https://registry.yarnpkg.com/amdefine/-/amdefine-1.0.1.tgz#4a5282ac164729e93619bcfd3ad151f817ce91f5" 28 | 29 | ansi-regex@^2.0.0: 30 | version "2.1.1" 31 | resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-2.1.1.tgz#c3b33ab5ee360d86e0e628f0468ae7ef27d654df" 32 | 33 | ansi-styles@^2.2.1: 34 | version "2.2.1" 35 | resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-2.2.1.tgz#b432dd3358b634cf75e1e4664368240533c1ddbe" 36 | 37 | ansi-styles@^3.2.1: 38 | version "3.2.1" 39 | resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-3.2.1.tgz#41fbb20243e50b12be0f04b8dedbf07520ce841d" 40 | dependencies: 41 | color-convert "^1.9.0" 42 | 43 | argparse@^1.0.2: 44 | version "1.0.10" 45 | resolved "https://registry.yarnpkg.com/argparse/-/argparse-1.0.10.tgz#bcd6791ea5ae09725e17e5ad988134cd40b3d911" 46 | dependencies: 47 | sprintf-js "~1.0.2" 48 | 49 | array-differ@^1.0.0: 50 | version "1.0.0" 51 | resolved "https://registry.yarnpkg.com/array-differ/-/array-differ-1.0.0.tgz#eff52e3758249d33be402b8bb8e564bb2b5d4031" 52 | 53 | array-find-index@^1.0.1: 54 | version "1.0.2" 55 | resolved "https://registry.yarnpkg.com/array-find-index/-/array-find-index-1.0.2.tgz#df010aa1287e164bbda6f9723b0a96a1ec4187a1" 56 | 57 | array-union@^1.0.1: 58 | version "1.0.2" 59 | resolved "https://registry.yarnpkg.com/array-union/-/array-union-1.0.2.tgz#9a34410e4f4e3da23dea375be5be70f24778ec39" 60 | dependencies: 61 | array-uniq "^1.0.1" 62 | 63 | array-uniq@^1.0.1: 64 | version "1.0.3" 65 | resolved "https://registry.yarnpkg.com/array-uniq/-/array-uniq-1.0.3.tgz#af6ac877a25cc7f74e058894753858dfdb24fdb6" 66 | 67 | arrify@^1.0.0: 68 | version "1.0.1" 69 | resolved "https://registry.yarnpkg.com/arrify/-/arrify-1.0.1.tgz#898508da2226f380df904728456849c1501a4b0d" 70 | 71 | async@^1.5.2, async@~1.5.2: 72 | version "1.5.2" 73 | resolved "https://registry.yarnpkg.com/async/-/async-1.5.2.tgz#ec6a61ae56480c0c3cb241c95618e20892f9672a" 74 | 75 | async@^2.6.0: 76 | version "2.6.1" 77 | resolved "https://registry.yarnpkg.com/async/-/async-2.6.1.tgz#b245a23ca71930044ec53fa46aa00a3e87c6a610" 78 | dependencies: 79 | lodash "^4.17.10" 80 | 81 | async@~0.2.6: 82 | version "0.2.10" 83 | resolved "https://registry.yarnpkg.com/async/-/async-0.2.10.tgz#b6bbe0b0674b9d719708ca38de8c237cb526c3d1" 84 | 85 | babel-code-frame@^6.22.0: 86 | version "6.22.0" 87 | resolved "https://registry.yarnpkg.com/babel-code-frame/-/babel-code-frame-6.22.0.tgz#027620bee567a88c32561574e7fd0801d33118e4" 88 | dependencies: 89 | chalk "^1.1.0" 90 | esutils "^2.0.2" 91 | js-tokens "^3.0.0" 92 | 93 | babel-code-frame@^6.26.0: 94 | version "6.26.0" 95 | resolved "https://registry.yarnpkg.com/babel-code-frame/-/babel-code-frame-6.26.0.tgz#63fd43f7dc1e3bb7ce35947db8fe369a3f58c74b" 96 | dependencies: 97 | chalk "^1.1.3" 98 | esutils "^2.0.2" 99 | js-tokens "^3.0.2" 100 | 101 | babel-core@^6.0.12, babel-core@^6.23.0: 102 | version "6.23.1" 103 | resolved "https://registry.yarnpkg.com/babel-core/-/babel-core-6.23.1.tgz#c143cb621bb2f621710c220c5d579d15b8a442df" 104 | dependencies: 105 | babel-code-frame "^6.22.0" 106 | babel-generator "^6.23.0" 107 | babel-helpers "^6.23.0" 108 | babel-messages "^6.23.0" 109 | babel-register "^6.23.0" 110 | babel-runtime "^6.22.0" 111 | babel-template "^6.23.0" 112 | babel-traverse "^6.23.1" 113 | babel-types "^6.23.0" 114 | babylon "^6.11.0" 115 | convert-source-map "^1.1.0" 116 | debug "^2.1.1" 117 | json5 "^0.5.0" 118 | lodash "^4.2.0" 119 | minimatch "^3.0.2" 120 | path-is-absolute "^1.0.0" 121 | private "^0.1.6" 122 | slash "^1.0.0" 123 | source-map "^0.5.0" 124 | 125 | babel-core@^6.24.1, babel-core@^6.26.0: 126 | version "6.26.3" 127 | resolved "https://registry.yarnpkg.com/babel-core/-/babel-core-6.26.3.tgz#b2e2f09e342d0f0c88e2f02e067794125e75c207" 128 | dependencies: 129 | babel-code-frame "^6.26.0" 130 | babel-generator "^6.26.0" 131 | babel-helpers "^6.24.1" 132 | babel-messages "^6.23.0" 133 | babel-register "^6.26.0" 134 | babel-runtime "^6.26.0" 135 | babel-template "^6.26.0" 136 | babel-traverse "^6.26.0" 137 | babel-types "^6.26.0" 138 | babylon "^6.18.0" 139 | convert-source-map "^1.5.1" 140 | debug "^2.6.9" 141 | json5 "^0.5.1" 142 | lodash "^4.17.4" 143 | minimatch "^3.0.4" 144 | path-is-absolute "^1.0.1" 145 | private "^0.1.8" 146 | slash "^1.0.0" 147 | source-map "^0.5.7" 148 | 149 | babel-generator@^6.23.0: 150 | version "6.23.0" 151 | resolved "https://registry.yarnpkg.com/babel-generator/-/babel-generator-6.23.0.tgz#6b8edab956ef3116f79d8c84c5a3c05f32a74bc5" 152 | dependencies: 153 | babel-messages "^6.23.0" 154 | babel-runtime "^6.22.0" 155 | babel-types "^6.23.0" 156 | detect-indent "^4.0.0" 157 | jsesc "^1.3.0" 158 | lodash "^4.2.0" 159 | source-map "^0.5.0" 160 | trim-right "^1.0.1" 161 | 162 | babel-generator@^6.26.0: 163 | version "6.26.1" 164 | resolved "https://registry.yarnpkg.com/babel-generator/-/babel-generator-6.26.1.tgz#1844408d3b8f0d35a404ea7ac180f087a601bd90" 165 | dependencies: 166 | babel-messages "^6.23.0" 167 | babel-runtime "^6.26.0" 168 | babel-types "^6.26.0" 169 | detect-indent "^4.0.0" 170 | jsesc "^1.3.0" 171 | lodash "^4.17.4" 172 | source-map "^0.5.7" 173 | trim-right "^1.0.1" 174 | 175 | babel-helper-builder-binary-assignment-operator-visitor@^6.24.1: 176 | version "6.24.1" 177 | resolved "https://registry.yarnpkg.com/babel-helper-builder-binary-assignment-operator-visitor/-/babel-helper-builder-binary-assignment-operator-visitor-6.24.1.tgz#cce4517ada356f4220bcae8a02c2b346f9a56664" 178 | dependencies: 179 | babel-helper-explode-assignable-expression "^6.24.1" 180 | babel-runtime "^6.22.0" 181 | babel-types "^6.24.1" 182 | 183 | babel-helper-call-delegate@^6.24.1: 184 | version "6.24.1" 185 | resolved "https://registry.yarnpkg.com/babel-helper-call-delegate/-/babel-helper-call-delegate-6.24.1.tgz#ece6aacddc76e41c3461f88bfc575bd0daa2df8d" 186 | dependencies: 187 | babel-helper-hoist-variables "^6.24.1" 188 | babel-runtime "^6.22.0" 189 | babel-traverse "^6.24.1" 190 | babel-types "^6.24.1" 191 | 192 | babel-helper-define-map@^6.24.1: 193 | version "6.26.0" 194 | resolved "https://registry.yarnpkg.com/babel-helper-define-map/-/babel-helper-define-map-6.26.0.tgz#a5f56dab41a25f97ecb498c7ebaca9819f95be5f" 195 | dependencies: 196 | babel-helper-function-name "^6.24.1" 197 | babel-runtime "^6.26.0" 198 | babel-types "^6.26.0" 199 | lodash "^4.17.4" 200 | 201 | babel-helper-explode-assignable-expression@^6.24.1: 202 | version "6.24.1" 203 | resolved "https://registry.yarnpkg.com/babel-helper-explode-assignable-expression/-/babel-helper-explode-assignable-expression-6.24.1.tgz#f25b82cf7dc10433c55f70592d5746400ac22caa" 204 | dependencies: 205 | babel-runtime "^6.22.0" 206 | babel-traverse "^6.24.1" 207 | babel-types "^6.24.1" 208 | 209 | babel-helper-function-name@^6.24.1: 210 | version "6.24.1" 211 | resolved "https://registry.yarnpkg.com/babel-helper-function-name/-/babel-helper-function-name-6.24.1.tgz#d3475b8c03ed98242a25b48351ab18399d3580a9" 212 | dependencies: 213 | babel-helper-get-function-arity "^6.24.1" 214 | babel-runtime "^6.22.0" 215 | babel-template "^6.24.1" 216 | babel-traverse "^6.24.1" 217 | babel-types "^6.24.1" 218 | 219 | babel-helper-get-function-arity@^6.24.1: 220 | version "6.24.1" 221 | resolved "https://registry.yarnpkg.com/babel-helper-get-function-arity/-/babel-helper-get-function-arity-6.24.1.tgz#8f7782aa93407c41d3aa50908f89b031b1b6853d" 222 | dependencies: 223 | babel-runtime "^6.22.0" 224 | babel-types "^6.24.1" 225 | 226 | babel-helper-hoist-variables@^6.22.0: 227 | version "6.22.0" 228 | resolved "https://registry.yarnpkg.com/babel-helper-hoist-variables/-/babel-helper-hoist-variables-6.22.0.tgz#3eacbf731d80705845dd2e9718f600cfb9b4ba72" 229 | dependencies: 230 | babel-runtime "^6.22.0" 231 | babel-types "^6.22.0" 232 | 233 | babel-helper-hoist-variables@^6.24.1: 234 | version "6.24.1" 235 | resolved "https://registry.yarnpkg.com/babel-helper-hoist-variables/-/babel-helper-hoist-variables-6.24.1.tgz#1ecb27689c9d25513eadbc9914a73f5408be7a76" 236 | dependencies: 237 | babel-runtime "^6.22.0" 238 | babel-types "^6.24.1" 239 | 240 | babel-helper-optimise-call-expression@^6.24.1: 241 | version "6.24.1" 242 | resolved "https://registry.yarnpkg.com/babel-helper-optimise-call-expression/-/babel-helper-optimise-call-expression-6.24.1.tgz#f7a13427ba9f73f8f4fa993c54a97882d1244257" 243 | dependencies: 244 | babel-runtime "^6.22.0" 245 | babel-types "^6.24.1" 246 | 247 | babel-helper-regex@^6.24.1: 248 | version "6.26.0" 249 | resolved "https://registry.yarnpkg.com/babel-helper-regex/-/babel-helper-regex-6.26.0.tgz#325c59f902f82f24b74faceed0363954f6495e72" 250 | dependencies: 251 | babel-runtime "^6.26.0" 252 | babel-types "^6.26.0" 253 | lodash "^4.17.4" 254 | 255 | babel-helper-remap-async-to-generator@^6.24.1: 256 | version "6.24.1" 257 | resolved "https://registry.yarnpkg.com/babel-helper-remap-async-to-generator/-/babel-helper-remap-async-to-generator-6.24.1.tgz#5ec581827ad723fecdd381f1c928390676e4551b" 258 | dependencies: 259 | babel-helper-function-name "^6.24.1" 260 | babel-runtime "^6.22.0" 261 | babel-template "^6.24.1" 262 | babel-traverse "^6.24.1" 263 | babel-types "^6.24.1" 264 | 265 | babel-helper-replace-supers@^6.24.1: 266 | version "6.24.1" 267 | resolved "https://registry.yarnpkg.com/babel-helper-replace-supers/-/babel-helper-replace-supers-6.24.1.tgz#bf6dbfe43938d17369a213ca8a8bf74b6a90ab1a" 268 | dependencies: 269 | babel-helper-optimise-call-expression "^6.24.1" 270 | babel-messages "^6.23.0" 271 | babel-runtime "^6.22.0" 272 | babel-template "^6.24.1" 273 | babel-traverse "^6.24.1" 274 | babel-types "^6.24.1" 275 | 276 | babel-helpers@^6.23.0: 277 | version "6.23.0" 278 | resolved "https://registry.yarnpkg.com/babel-helpers/-/babel-helpers-6.23.0.tgz#4f8f2e092d0b6a8808a4bde79c27f1e2ecf0d992" 279 | dependencies: 280 | babel-runtime "^6.22.0" 281 | babel-template "^6.23.0" 282 | 283 | babel-helpers@^6.24.1: 284 | version "6.24.1" 285 | resolved "https://registry.yarnpkg.com/babel-helpers/-/babel-helpers-6.24.1.tgz#3471de9caec388e5c850e597e58a26ddf37602b2" 286 | dependencies: 287 | babel-runtime "^6.22.0" 288 | babel-template "^6.24.1" 289 | 290 | babel-messages@^6.23.0: 291 | version "6.23.0" 292 | resolved "https://registry.yarnpkg.com/babel-messages/-/babel-messages-6.23.0.tgz#f3cdf4703858035b2a2951c6ec5edf6c62f2630e" 293 | dependencies: 294 | babel-runtime "^6.22.0" 295 | 296 | babel-plugin-check-es2015-constants@^6.22.0: 297 | version "6.22.0" 298 | resolved "https://registry.yarnpkg.com/babel-plugin-check-es2015-constants/-/babel-plugin-check-es2015-constants-6.22.0.tgz#35157b101426fd2ffd3da3f75c7d1e91835bbf8a" 299 | dependencies: 300 | babel-runtime "^6.22.0" 301 | 302 | babel-plugin-syntax-async-functions@^6.8.0: 303 | version "6.13.0" 304 | resolved "https://registry.yarnpkg.com/babel-plugin-syntax-async-functions/-/babel-plugin-syntax-async-functions-6.13.0.tgz#cad9cad1191b5ad634bf30ae0872391e0647be95" 305 | 306 | babel-plugin-syntax-dynamic-import@^6.18.0: 307 | version "6.18.0" 308 | resolved "https://registry.yarnpkg.com/babel-plugin-syntax-dynamic-import/-/babel-plugin-syntax-dynamic-import-6.18.0.tgz#8d6a26229c83745a9982a441051572caa179b1da" 309 | 310 | babel-plugin-syntax-exponentiation-operator@^6.8.0: 311 | version "6.13.0" 312 | resolved "https://registry.yarnpkg.com/babel-plugin-syntax-exponentiation-operator/-/babel-plugin-syntax-exponentiation-operator-6.13.0.tgz#9ee7e8337290da95288201a6a57f4170317830de" 313 | 314 | babel-plugin-syntax-object-rest-spread@^6.8.0: 315 | version "6.13.0" 316 | resolved "https://registry.yarnpkg.com/babel-plugin-syntax-object-rest-spread/-/babel-plugin-syntax-object-rest-spread-6.13.0.tgz#fd6536f2bce13836ffa3a5458c4903a597bb3bf5" 317 | 318 | babel-plugin-syntax-trailing-function-commas@^6.22.0: 319 | version "6.22.0" 320 | resolved "https://registry.yarnpkg.com/babel-plugin-syntax-trailing-function-commas/-/babel-plugin-syntax-trailing-function-commas-6.22.0.tgz#ba0360937f8d06e40180a43fe0d5616fff532cf3" 321 | 322 | babel-plugin-transform-amd-system-wrapper@^0.3.7: 323 | version "0.3.7" 324 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-amd-system-wrapper/-/babel-plugin-transform-amd-system-wrapper-0.3.7.tgz#521c782d35644491c979ea683e8a5e1caff0ba42" 325 | dependencies: 326 | babel-template "^6.9.0" 327 | 328 | babel-plugin-transform-async-to-generator@^6.22.0: 329 | version "6.24.1" 330 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-async-to-generator/-/babel-plugin-transform-async-to-generator-6.24.1.tgz#6536e378aff6cb1d5517ac0e40eb3e9fc8d08761" 331 | dependencies: 332 | babel-helper-remap-async-to-generator "^6.24.1" 333 | babel-plugin-syntax-async-functions "^6.8.0" 334 | babel-runtime "^6.22.0" 335 | 336 | babel-plugin-transform-cjs-system-wrapper@^0.6.2: 337 | version "0.6.2" 338 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-cjs-system-wrapper/-/babel-plugin-transform-cjs-system-wrapper-0.6.2.tgz#bd7494775289424ff493b6ed455de495bd71ba1d" 339 | dependencies: 340 | babel-template "^6.9.0" 341 | 342 | babel-plugin-transform-es2015-arrow-functions@^6.22.0: 343 | version "6.22.0" 344 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-arrow-functions/-/babel-plugin-transform-es2015-arrow-functions-6.22.0.tgz#452692cb711d5f79dc7f85e440ce41b9f244d221" 345 | dependencies: 346 | babel-runtime "^6.22.0" 347 | 348 | babel-plugin-transform-es2015-block-scoped-functions@^6.22.0: 349 | version "6.22.0" 350 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-block-scoped-functions/-/babel-plugin-transform-es2015-block-scoped-functions-6.22.0.tgz#bbc51b49f964d70cb8d8e0b94e820246ce3a6141" 351 | dependencies: 352 | babel-runtime "^6.22.0" 353 | 354 | babel-plugin-transform-es2015-block-scoping@^6.23.0: 355 | version "6.26.0" 356 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-block-scoping/-/babel-plugin-transform-es2015-block-scoping-6.26.0.tgz#d70f5299c1308d05c12f463813b0a09e73b1895f" 357 | dependencies: 358 | babel-runtime "^6.26.0" 359 | babel-template "^6.26.0" 360 | babel-traverse "^6.26.0" 361 | babel-types "^6.26.0" 362 | lodash "^4.17.4" 363 | 364 | babel-plugin-transform-es2015-classes@^6.23.0: 365 | version "6.24.1" 366 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-classes/-/babel-plugin-transform-es2015-classes-6.24.1.tgz#5a4c58a50c9c9461e564b4b2a3bfabc97a2584db" 367 | dependencies: 368 | babel-helper-define-map "^6.24.1" 369 | babel-helper-function-name "^6.24.1" 370 | babel-helper-optimise-call-expression "^6.24.1" 371 | babel-helper-replace-supers "^6.24.1" 372 | babel-messages "^6.23.0" 373 | babel-runtime "^6.22.0" 374 | babel-template "^6.24.1" 375 | babel-traverse "^6.24.1" 376 | babel-types "^6.24.1" 377 | 378 | babel-plugin-transform-es2015-computed-properties@^6.22.0: 379 | version "6.24.1" 380 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-computed-properties/-/babel-plugin-transform-es2015-computed-properties-6.24.1.tgz#6fe2a8d16895d5634f4cd999b6d3480a308159b3" 381 | dependencies: 382 | babel-runtime "^6.22.0" 383 | babel-template "^6.24.1" 384 | 385 | babel-plugin-transform-es2015-destructuring@^6.23.0: 386 | version "6.23.0" 387 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-destructuring/-/babel-plugin-transform-es2015-destructuring-6.23.0.tgz#997bb1f1ab967f682d2b0876fe358d60e765c56d" 388 | dependencies: 389 | babel-runtime "^6.22.0" 390 | 391 | babel-plugin-transform-es2015-duplicate-keys@^6.22.0: 392 | version "6.24.1" 393 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-duplicate-keys/-/babel-plugin-transform-es2015-duplicate-keys-6.24.1.tgz#73eb3d310ca969e3ef9ec91c53741a6f1576423e" 394 | dependencies: 395 | babel-runtime "^6.22.0" 396 | babel-types "^6.24.1" 397 | 398 | babel-plugin-transform-es2015-for-of@^6.23.0: 399 | version "6.23.0" 400 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-for-of/-/babel-plugin-transform-es2015-for-of-6.23.0.tgz#f47c95b2b613df1d3ecc2fdb7573623c75248691" 401 | dependencies: 402 | babel-runtime "^6.22.0" 403 | 404 | babel-plugin-transform-es2015-function-name@^6.22.0: 405 | version "6.24.1" 406 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-function-name/-/babel-plugin-transform-es2015-function-name-6.24.1.tgz#834c89853bc36b1af0f3a4c5dbaa94fd8eacaa8b" 407 | dependencies: 408 | babel-helper-function-name "^6.24.1" 409 | babel-runtime "^6.22.0" 410 | babel-types "^6.24.1" 411 | 412 | babel-plugin-transform-es2015-literals@^6.22.0: 413 | version "6.22.0" 414 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-literals/-/babel-plugin-transform-es2015-literals-6.22.0.tgz#4f54a02d6cd66cf915280019a31d31925377ca2e" 415 | dependencies: 416 | babel-runtime "^6.22.0" 417 | 418 | babel-plugin-transform-es2015-modules-amd@^6.22.0, babel-plugin-transform-es2015-modules-amd@^6.24.1: 419 | version "6.24.1" 420 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-modules-amd/-/babel-plugin-transform-es2015-modules-amd-6.24.1.tgz#3b3e54017239842d6d19c3011c4bd2f00a00d154" 421 | dependencies: 422 | babel-plugin-transform-es2015-modules-commonjs "^6.24.1" 423 | babel-runtime "^6.22.0" 424 | babel-template "^6.24.1" 425 | 426 | babel-plugin-transform-es2015-modules-commonjs@^6.23.0, babel-plugin-transform-es2015-modules-commonjs@^6.24.1: 427 | version "6.26.2" 428 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-modules-commonjs/-/babel-plugin-transform-es2015-modules-commonjs-6.26.2.tgz#58a793863a9e7ca870bdc5a881117ffac27db6f3" 429 | dependencies: 430 | babel-plugin-transform-strict-mode "^6.24.1" 431 | babel-runtime "^6.26.0" 432 | babel-template "^6.26.0" 433 | babel-types "^6.26.0" 434 | 435 | babel-plugin-transform-es2015-modules-systemjs@^6.23.0: 436 | version "6.24.1" 437 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-modules-systemjs/-/babel-plugin-transform-es2015-modules-systemjs-6.24.1.tgz#ff89a142b9119a906195f5f106ecf305d9407d23" 438 | dependencies: 439 | babel-helper-hoist-variables "^6.24.1" 440 | babel-runtime "^6.22.0" 441 | babel-template "^6.24.1" 442 | 443 | babel-plugin-transform-es2015-modules-systemjs@^6.6.5: 444 | version "6.23.0" 445 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-modules-systemjs/-/babel-plugin-transform-es2015-modules-systemjs-6.23.0.tgz#ae3469227ffac39b0310d90fec73bfdc4f6317b0" 446 | dependencies: 447 | babel-helper-hoist-variables "^6.22.0" 448 | babel-runtime "^6.22.0" 449 | babel-template "^6.23.0" 450 | 451 | babel-plugin-transform-es2015-modules-umd@^6.23.0: 452 | version "6.24.1" 453 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-modules-umd/-/babel-plugin-transform-es2015-modules-umd-6.24.1.tgz#ac997e6285cd18ed6176adb607d602344ad38468" 454 | dependencies: 455 | babel-plugin-transform-es2015-modules-amd "^6.24.1" 456 | babel-runtime "^6.22.0" 457 | babel-template "^6.24.1" 458 | 459 | babel-plugin-transform-es2015-object-super@^6.22.0: 460 | version "6.24.1" 461 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-object-super/-/babel-plugin-transform-es2015-object-super-6.24.1.tgz#24cef69ae21cb83a7f8603dad021f572eb278f8d" 462 | dependencies: 463 | babel-helper-replace-supers "^6.24.1" 464 | babel-runtime "^6.22.0" 465 | 466 | babel-plugin-transform-es2015-parameters@^6.23.0: 467 | version "6.24.1" 468 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-parameters/-/babel-plugin-transform-es2015-parameters-6.24.1.tgz#57ac351ab49caf14a97cd13b09f66fdf0a625f2b" 469 | dependencies: 470 | babel-helper-call-delegate "^6.24.1" 471 | babel-helper-get-function-arity "^6.24.1" 472 | babel-runtime "^6.22.0" 473 | babel-template "^6.24.1" 474 | babel-traverse "^6.24.1" 475 | babel-types "^6.24.1" 476 | 477 | babel-plugin-transform-es2015-shorthand-properties@^6.22.0: 478 | version "6.24.1" 479 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-shorthand-properties/-/babel-plugin-transform-es2015-shorthand-properties-6.24.1.tgz#24f875d6721c87661bbd99a4622e51f14de38aa0" 480 | dependencies: 481 | babel-runtime "^6.22.0" 482 | babel-types "^6.24.1" 483 | 484 | babel-plugin-transform-es2015-spread@^6.22.0: 485 | version "6.22.0" 486 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-spread/-/babel-plugin-transform-es2015-spread-6.22.0.tgz#d6d68a99f89aedc4536c81a542e8dd9f1746f8d1" 487 | dependencies: 488 | babel-runtime "^6.22.0" 489 | 490 | babel-plugin-transform-es2015-sticky-regex@^6.22.0: 491 | version "6.24.1" 492 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-sticky-regex/-/babel-plugin-transform-es2015-sticky-regex-6.24.1.tgz#00c1cdb1aca71112cdf0cf6126c2ed6b457ccdbc" 493 | dependencies: 494 | babel-helper-regex "^6.24.1" 495 | babel-runtime "^6.22.0" 496 | babel-types "^6.24.1" 497 | 498 | babel-plugin-transform-es2015-template-literals@^6.22.0: 499 | version "6.22.0" 500 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-template-literals/-/babel-plugin-transform-es2015-template-literals-6.22.0.tgz#a84b3450f7e9f8f1f6839d6d687da84bb1236d8d" 501 | dependencies: 502 | babel-runtime "^6.22.0" 503 | 504 | babel-plugin-transform-es2015-typeof-symbol@^6.23.0: 505 | version "6.23.0" 506 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-typeof-symbol/-/babel-plugin-transform-es2015-typeof-symbol-6.23.0.tgz#dec09f1cddff94b52ac73d505c84df59dcceb372" 507 | dependencies: 508 | babel-runtime "^6.22.0" 509 | 510 | babel-plugin-transform-es2015-unicode-regex@^6.22.0: 511 | version "6.24.1" 512 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-es2015-unicode-regex/-/babel-plugin-transform-es2015-unicode-regex-6.24.1.tgz#d38b12f42ea7323f729387f18a7c5ae1faeb35e9" 513 | dependencies: 514 | babel-helper-regex "^6.24.1" 515 | babel-runtime "^6.22.0" 516 | regexpu-core "^2.0.0" 517 | 518 | babel-plugin-transform-exponentiation-operator@^6.22.0: 519 | version "6.24.1" 520 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-exponentiation-operator/-/babel-plugin-transform-exponentiation-operator-6.24.1.tgz#2ab0c9c7f3098fa48907772bb813fe41e8de3a0e" 521 | dependencies: 522 | babel-helper-builder-binary-assignment-operator-visitor "^6.24.1" 523 | babel-plugin-syntax-exponentiation-operator "^6.8.0" 524 | babel-runtime "^6.22.0" 525 | 526 | babel-plugin-transform-global-system-wrapper@^0.3.4: 527 | version "0.3.4" 528 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-global-system-wrapper/-/babel-plugin-transform-global-system-wrapper-0.3.4.tgz#948dd7d29fc21447e39bd3447f2debc7f2f73aac" 529 | dependencies: 530 | babel-template "^6.9.0" 531 | 532 | babel-plugin-transform-object-rest-spread@^6.26.0: 533 | version "6.26.0" 534 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-object-rest-spread/-/babel-plugin-transform-object-rest-spread-6.26.0.tgz#0f36692d50fef6b7e2d4b3ac1478137a963b7b06" 535 | dependencies: 536 | babel-plugin-syntax-object-rest-spread "^6.8.0" 537 | babel-runtime "^6.26.0" 538 | 539 | babel-plugin-transform-regenerator@^6.22.0: 540 | version "6.26.0" 541 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-regenerator/-/babel-plugin-transform-regenerator-6.26.0.tgz#e0703696fbde27f0a3efcacf8b4dca2f7b3a8f2f" 542 | dependencies: 543 | regenerator-transform "^0.10.0" 544 | 545 | babel-plugin-transform-strict-mode@^6.24.1: 546 | version "6.24.1" 547 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-strict-mode/-/babel-plugin-transform-strict-mode-6.24.1.tgz#d5faf7aa578a65bbe591cf5edae04a0c67020758" 548 | dependencies: 549 | babel-runtime "^6.22.0" 550 | babel-types "^6.24.1" 551 | 552 | babel-plugin-transform-system-register@^0.0.1: 553 | version "0.0.1" 554 | resolved "https://registry.yarnpkg.com/babel-plugin-transform-system-register/-/babel-plugin-transform-system-register-0.0.1.tgz#9dff40390c2763ac518f0b2ad7c5ea4f65a5be25" 555 | 556 | babel-preset-env@^1.7.0: 557 | version "1.7.0" 558 | resolved "https://registry.yarnpkg.com/babel-preset-env/-/babel-preset-env-1.7.0.tgz#dea79fa4ebeb883cd35dab07e260c1c9c04df77a" 559 | dependencies: 560 | babel-plugin-check-es2015-constants "^6.22.0" 561 | babel-plugin-syntax-trailing-function-commas "^6.22.0" 562 | babel-plugin-transform-async-to-generator "^6.22.0" 563 | babel-plugin-transform-es2015-arrow-functions "^6.22.0" 564 | babel-plugin-transform-es2015-block-scoped-functions "^6.22.0" 565 | babel-plugin-transform-es2015-block-scoping "^6.23.0" 566 | babel-plugin-transform-es2015-classes "^6.23.0" 567 | babel-plugin-transform-es2015-computed-properties "^6.22.0" 568 | babel-plugin-transform-es2015-destructuring "^6.23.0" 569 | babel-plugin-transform-es2015-duplicate-keys "^6.22.0" 570 | babel-plugin-transform-es2015-for-of "^6.23.0" 571 | babel-plugin-transform-es2015-function-name "^6.22.0" 572 | babel-plugin-transform-es2015-literals "^6.22.0" 573 | babel-plugin-transform-es2015-modules-amd "^6.22.0" 574 | babel-plugin-transform-es2015-modules-commonjs "^6.23.0" 575 | babel-plugin-transform-es2015-modules-systemjs "^6.23.0" 576 | babel-plugin-transform-es2015-modules-umd "^6.23.0" 577 | babel-plugin-transform-es2015-object-super "^6.22.0" 578 | babel-plugin-transform-es2015-parameters "^6.23.0" 579 | babel-plugin-transform-es2015-shorthand-properties "^6.22.0" 580 | babel-plugin-transform-es2015-spread "^6.22.0" 581 | babel-plugin-transform-es2015-sticky-regex "^6.22.0" 582 | babel-plugin-transform-es2015-template-literals "^6.22.0" 583 | babel-plugin-transform-es2015-typeof-symbol "^6.23.0" 584 | babel-plugin-transform-es2015-unicode-regex "^6.22.0" 585 | babel-plugin-transform-exponentiation-operator "^6.22.0" 586 | babel-plugin-transform-regenerator "^6.22.0" 587 | browserslist "^3.2.6" 588 | invariant "^2.2.2" 589 | semver "^5.3.0" 590 | 591 | babel-register@^6.23.0: 592 | version "6.23.0" 593 | resolved "https://registry.yarnpkg.com/babel-register/-/babel-register-6.23.0.tgz#c9aa3d4cca94b51da34826c4a0f9e08145d74ff3" 594 | dependencies: 595 | babel-core "^6.23.0" 596 | babel-runtime "^6.22.0" 597 | core-js "^2.4.0" 598 | home-or-tmp "^2.0.0" 599 | lodash "^4.2.0" 600 | mkdirp "^0.5.1" 601 | source-map-support "^0.4.2" 602 | 603 | babel-register@^6.26.0: 604 | version "6.26.0" 605 | resolved "https://registry.yarnpkg.com/babel-register/-/babel-register-6.26.0.tgz#6ed021173e2fcb486d7acb45c6009a856f647071" 606 | dependencies: 607 | babel-core "^6.26.0" 608 | babel-runtime "^6.26.0" 609 | core-js "^2.5.0" 610 | home-or-tmp "^2.0.0" 611 | lodash "^4.17.4" 612 | mkdirp "^0.5.1" 613 | source-map-support "^0.4.15" 614 | 615 | babel-runtime@^6.18.0, babel-runtime@^6.22.0: 616 | version "6.23.0" 617 | resolved "https://registry.yarnpkg.com/babel-runtime/-/babel-runtime-6.23.0.tgz#0a9489f144de70efb3ce4300accdb329e2fc543b" 618 | dependencies: 619 | core-js "^2.4.0" 620 | regenerator-runtime "^0.10.0" 621 | 622 | babel-runtime@^6.26.0: 623 | version "6.26.0" 624 | resolved "https://registry.yarnpkg.com/babel-runtime/-/babel-runtime-6.26.0.tgz#965c7058668e82b55d7bfe04ff2337bc8b5647fe" 625 | dependencies: 626 | core-js "^2.4.0" 627 | regenerator-runtime "^0.11.0" 628 | 629 | babel-template@^6.23.0, babel-template@^6.9.0: 630 | version "6.23.0" 631 | resolved "https://registry.yarnpkg.com/babel-template/-/babel-template-6.23.0.tgz#04d4f270adbb3aa704a8143ae26faa529238e638" 632 | dependencies: 633 | babel-runtime "^6.22.0" 634 | babel-traverse "^6.23.0" 635 | babel-types "^6.23.0" 636 | babylon "^6.11.0" 637 | lodash "^4.2.0" 638 | 639 | babel-template@^6.24.1, babel-template@^6.26.0: 640 | version "6.26.0" 641 | resolved "https://registry.yarnpkg.com/babel-template/-/babel-template-6.26.0.tgz#de03e2d16396b069f46dd9fff8521fb1a0e35e02" 642 | dependencies: 643 | babel-runtime "^6.26.0" 644 | babel-traverse "^6.26.0" 645 | babel-types "^6.26.0" 646 | babylon "^6.18.0" 647 | lodash "^4.17.4" 648 | 649 | babel-traverse@^6.23.0, babel-traverse@^6.23.1: 650 | version "6.23.1" 651 | resolved "https://registry.yarnpkg.com/babel-traverse/-/babel-traverse-6.23.1.tgz#d3cb59010ecd06a97d81310065f966b699e14f48" 652 | dependencies: 653 | babel-code-frame "^6.22.0" 654 | babel-messages "^6.23.0" 655 | babel-runtime "^6.22.0" 656 | babel-types "^6.23.0" 657 | babylon "^6.15.0" 658 | debug "^2.2.0" 659 | globals "^9.0.0" 660 | invariant "^2.2.0" 661 | lodash "^4.2.0" 662 | 663 | babel-traverse@^6.24.1, babel-traverse@^6.26.0: 664 | version "6.26.0" 665 | resolved "https://registry.yarnpkg.com/babel-traverse/-/babel-traverse-6.26.0.tgz#46a9cbd7edcc62c8e5c064e2d2d8d0f4035766ee" 666 | dependencies: 667 | babel-code-frame "^6.26.0" 668 | babel-messages "^6.23.0" 669 | babel-runtime "^6.26.0" 670 | babel-types "^6.26.0" 671 | babylon "^6.18.0" 672 | debug "^2.6.8" 673 | globals "^9.18.0" 674 | invariant "^2.2.2" 675 | lodash "^4.17.4" 676 | 677 | babel-types@^6.19.0, babel-types@^6.22.0, babel-types@^6.23.0: 678 | version "6.23.0" 679 | resolved "https://registry.yarnpkg.com/babel-types/-/babel-types-6.23.0.tgz#bb17179d7538bad38cd0c9e115d340f77e7e9acf" 680 | dependencies: 681 | babel-runtime "^6.22.0" 682 | esutils "^2.0.2" 683 | lodash "^4.2.0" 684 | to-fast-properties "^1.0.1" 685 | 686 | babel-types@^6.24.1, babel-types@^6.26.0: 687 | version "6.26.0" 688 | resolved "https://registry.yarnpkg.com/babel-types/-/babel-types-6.26.0.tgz#a3b073f94ab49eb6fa55cd65227a334380632497" 689 | dependencies: 690 | babel-runtime "^6.26.0" 691 | esutils "^2.0.2" 692 | lodash "^4.17.4" 693 | to-fast-properties "^1.0.3" 694 | 695 | babel@^6.23.0: 696 | version "6.23.0" 697 | resolved "https://registry.yarnpkg.com/babel/-/babel-6.23.0.tgz#d0d1e7d803e974765beea3232d4e153c0efb90f4" 698 | 699 | babylon@^6.11.0, babylon@^6.15.0: 700 | version "6.15.0" 701 | resolved "https://registry.yarnpkg.com/babylon/-/babylon-6.15.0.tgz#ba65cfa1a80e1759b0e89fb562e27dccae70348e" 702 | 703 | babylon@^6.18.0: 704 | version "6.18.0" 705 | resolved "https://registry.yarnpkg.com/babylon/-/babylon-6.18.0.tgz#af2f3b88fa6f5c1e4c634d1a0f8eac4f55b395e3" 706 | 707 | balanced-match@^0.4.1: 708 | version "0.4.2" 709 | resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-0.4.2.tgz#cb3f3e3c732dc0f01ee70b403f302e61d7709838" 710 | 711 | balanced-match@^1.0.0: 712 | version "1.0.0" 713 | resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.0.tgz#89b4d199ab2bee49de164ea02b89ce462d71b767" 714 | 715 | bluebird@^3.3.4: 716 | version "3.4.7" 717 | resolved "https://registry.yarnpkg.com/bluebird/-/bluebird-3.4.7.tgz#f72d760be09b7f76d08ed8fae98b289a8d05fab3" 718 | 719 | body@^5.1.0: 720 | version "5.1.0" 721 | resolved "https://registry.yarnpkg.com/body/-/body-5.1.0.tgz#e4ba0ce410a46936323367609ecb4e6553125069" 722 | dependencies: 723 | continuable-cache "^0.3.1" 724 | error "^7.0.0" 725 | raw-body "~1.1.0" 726 | safe-json-parse "~1.0.1" 727 | 728 | brace-expansion@^1.0.0: 729 | version "1.1.6" 730 | resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.6.tgz#7197d7eaa9b87e648390ea61fc66c84427420df9" 731 | dependencies: 732 | balanced-match "^0.4.1" 733 | concat-map "0.0.1" 734 | 735 | brace-expansion@^1.1.7: 736 | version "1.1.11" 737 | resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd" 738 | dependencies: 739 | balanced-match "^1.0.0" 740 | concat-map "0.0.1" 741 | 742 | browserify-zlib@^0.1.4: 743 | version "0.1.4" 744 | resolved "https://registry.yarnpkg.com/browserify-zlib/-/browserify-zlib-0.1.4.tgz#bb35f8a519f600e0fa6b8485241c979d0141fb2d" 745 | dependencies: 746 | pako "~0.2.0" 747 | 748 | browserslist@^3.2.6: 749 | version "3.2.8" 750 | resolved "https://registry.yarnpkg.com/browserslist/-/browserslist-3.2.8.tgz#b0005361d6471f0f5952797a76fc985f1f978fc6" 751 | dependencies: 752 | caniuse-lite "^1.0.30000844" 753 | electron-to-chromium "^1.3.47" 754 | 755 | buffer-from@^1.0.0: 756 | version "1.1.0" 757 | resolved "https://registry.yarnpkg.com/buffer-from/-/buffer-from-1.1.0.tgz#87fcaa3a298358e0ade6e442cfce840740d1ad04" 758 | 759 | builtin-modules@^1.0.0: 760 | version "1.1.1" 761 | resolved "https://registry.yarnpkg.com/builtin-modules/-/builtin-modules-1.1.1.tgz#270f076c5a72c02f5b65a47df94c5fe3a278892f" 762 | 763 | bytes@1: 764 | version "1.0.0" 765 | resolved "https://registry.yarnpkg.com/bytes/-/bytes-1.0.0.tgz#3569ede8ba34315fab99c3e92cb04c7220de1fa8" 766 | 767 | camelcase-keys@^2.0.0: 768 | version "2.1.0" 769 | resolved "https://registry.yarnpkg.com/camelcase-keys/-/camelcase-keys-2.1.0.tgz#308beeaffdf28119051efa1d932213c91b8f92e7" 770 | dependencies: 771 | camelcase "^2.0.0" 772 | map-obj "^1.0.0" 773 | 774 | camelcase@^1.0.2: 775 | version "1.2.1" 776 | resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-1.2.1.tgz#9bb5304d2e0b56698b2c758b08a3eaa9daa58a39" 777 | 778 | camelcase@^2.0.0: 779 | version "2.1.1" 780 | resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-2.1.1.tgz#7c1d16d679a1bbe59ca02cacecfb011e201f5a1f" 781 | 782 | caniuse-lite@^1.0.30000844: 783 | version "1.0.30000859" 784 | resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30000859.tgz#da974adc5348fffe94724877a7ef8cb5d6d3d777" 785 | 786 | center-align@^0.1.1: 787 | version "0.1.3" 788 | resolved "https://registry.yarnpkg.com/center-align/-/center-align-0.1.3.tgz#aa0d32629b6ee972200411cbd4461c907bc2b7ad" 789 | dependencies: 790 | align-text "^0.1.3" 791 | lazy-cache "^1.0.3" 792 | 793 | chalk@^1.0.0, chalk@^1.1.0, chalk@^1.1.1, chalk@^1.1.3: 794 | version "1.1.3" 795 | resolved "https://registry.yarnpkg.com/chalk/-/chalk-1.1.3.tgz#a8115c55e4a702fe4d150abd3872822a7e09fc98" 796 | dependencies: 797 | ansi-styles "^2.2.1" 798 | escape-string-regexp "^1.0.2" 799 | has-ansi "^2.0.0" 800 | strip-ansi "^3.0.0" 801 | supports-color "^2.0.0" 802 | 803 | chalk@~2.4.1: 804 | version "2.4.1" 805 | resolved "https://registry.yarnpkg.com/chalk/-/chalk-2.4.1.tgz#18c49ab16a037b6eb0152cc83e3471338215b66e" 806 | dependencies: 807 | ansi-styles "^3.2.1" 808 | escape-string-regexp "^1.0.5" 809 | supports-color "^5.3.0" 810 | 811 | cliui@^2.1.0: 812 | version "2.1.0" 813 | resolved "https://registry.yarnpkg.com/cliui/-/cliui-2.1.0.tgz#4b475760ff80264c762c3a1719032e91c7fea0d1" 814 | dependencies: 815 | center-align "^0.1.1" 816 | right-align "^0.1.1" 817 | wordwrap "0.0.2" 818 | 819 | coffeescript@~1.10.0: 820 | version "1.10.0" 821 | resolved "https://registry.yarnpkg.com/coffeescript/-/coffeescript-1.10.0.tgz#e7aa8301917ef621b35d8a39f348dcdd1db7e33e" 822 | 823 | color-convert@^1.9.0: 824 | version "1.9.2" 825 | resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-1.9.2.tgz#49881b8fba67df12a96bdf3f56c0aab9e7913147" 826 | dependencies: 827 | color-name "1.1.1" 828 | 829 | color-name@1.1.1: 830 | version "1.1.1" 831 | resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.1.tgz#4b1415304cf50028ea81643643bd82ea05803689" 832 | 833 | colors@~1.1.2: 834 | version "1.1.2" 835 | resolved "https://registry.yarnpkg.com/colors/-/colors-1.1.2.tgz#168a4701756b6a7f51a12ce0c97bfa28c084ed63" 836 | 837 | commander@2.9.x: 838 | version "2.9.0" 839 | resolved "https://registry.yarnpkg.com/commander/-/commander-2.9.0.tgz#9c99094176e12240cb22d6c5146098400fe0f7d4" 840 | dependencies: 841 | graceful-readlink ">= 1.0.0" 842 | 843 | concat-map@0.0.1: 844 | version "0.0.1" 845 | resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b" 846 | 847 | concat-stream@^1.4.1: 848 | version "1.6.2" 849 | resolved "https://registry.yarnpkg.com/concat-stream/-/concat-stream-1.6.2.tgz#904bdf194cd3122fc675c77fc4ac3d4ff0fd1a34" 850 | dependencies: 851 | buffer-from "^1.0.0" 852 | inherits "^2.0.3" 853 | readable-stream "^2.2.2" 854 | typedarray "^0.0.6" 855 | 856 | continuable-cache@^0.3.1: 857 | version "0.3.1" 858 | resolved "https://registry.yarnpkg.com/continuable-cache/-/continuable-cache-0.3.1.tgz#bd727a7faed77e71ff3985ac93351a912733ad0f" 859 | 860 | convert-source-map@^1.1.0: 861 | version "1.4.0" 862 | resolved "https://registry.yarnpkg.com/convert-source-map/-/convert-source-map-1.4.0.tgz#e3dad195bf61bfe13a7a3c73e9876ec14a0268f3" 863 | 864 | convert-source-map@^1.5.1: 865 | version "1.5.1" 866 | resolved "https://registry.yarnpkg.com/convert-source-map/-/convert-source-map-1.5.1.tgz#b8278097b9bc229365de5c62cf5fcaed8b5599e5" 867 | 868 | core-js@^2.4.0: 869 | version "2.4.1" 870 | resolved "https://registry.yarnpkg.com/core-js/-/core-js-2.4.1.tgz#4de911e667b0eae9124e34254b53aea6fc618d3e" 871 | 872 | core-js@^2.5.0: 873 | version "2.5.7" 874 | resolved "https://registry.yarnpkg.com/core-js/-/core-js-2.5.7.tgz#f972608ff0cead68b841a16a932d0b183791814e" 875 | 876 | core-util-is@~1.0.0: 877 | version "1.0.2" 878 | resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.2.tgz#b5fd54220aa2bc5ab57aab7140c940754503c1a7" 879 | 880 | currently-unhandled@^0.4.1: 881 | version "0.4.1" 882 | resolved "https://registry.yarnpkg.com/currently-unhandled/-/currently-unhandled-0.4.1.tgz#988df33feab191ef799a61369dd76c17adf957ea" 883 | dependencies: 884 | array-find-index "^1.0.1" 885 | 886 | d@1: 887 | version "1.0.0" 888 | resolved "https://registry.yarnpkg.com/d/-/d-1.0.0.tgz#754bb5bfe55451da69a58b94d45f4c5b0462d58f" 889 | dependencies: 890 | es5-ext "^0.10.9" 891 | 892 | d@^0.1.1, d@~0.1.1: 893 | version "0.1.1" 894 | resolved "https://registry.yarnpkg.com/d/-/d-0.1.1.tgz#da184c535d18d8ee7ba2aa229b914009fae11309" 895 | dependencies: 896 | es5-ext "~0.10.2" 897 | 898 | data-uri-to-buffer@0.0.4: 899 | version "0.0.4" 900 | resolved "https://registry.yarnpkg.com/data-uri-to-buffer/-/data-uri-to-buffer-0.0.4.tgz#46e13ab9da8e309745c8d01ce547213ebdb2fe3f" 901 | 902 | dateformat@~1.0.12: 903 | version "1.0.12" 904 | resolved "https://registry.yarnpkg.com/dateformat/-/dateformat-1.0.12.tgz#9f124b67594c937ff706932e4a642cca8dbbfee9" 905 | dependencies: 906 | get-stdin "^4.0.1" 907 | meow "^3.3.0" 908 | 909 | debug@^2.1.1, debug@^2.2.0: 910 | version "2.2.0" 911 | resolved "https://registry.yarnpkg.com/debug/-/debug-2.2.0.tgz#f87057e995b1a1f6ae6a4960664137bc56f039da" 912 | dependencies: 913 | ms "0.7.1" 914 | 915 | debug@^2.6.8, debug@^2.6.9: 916 | version "2.6.9" 917 | resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.9.tgz#5d128515df134ff327e90a4c93f4e077a536341f" 918 | dependencies: 919 | ms "2.0.0" 920 | 921 | debug@^3.1.0: 922 | version "3.1.0" 923 | resolved "https://registry.yarnpkg.com/debug/-/debug-3.1.0.tgz#5bb5a0672628b64149566ba16819e61518c67261" 924 | dependencies: 925 | ms "2.0.0" 926 | 927 | decamelize@^1.0.0, decamelize@^1.1.2: 928 | version "1.2.0" 929 | resolved "https://registry.yarnpkg.com/decamelize/-/decamelize-1.2.0.tgz#f6534d15148269b20352e7bee26f501f9a191290" 930 | 931 | define-properties@^1.1.2: 932 | version "1.1.2" 933 | resolved "https://registry.yarnpkg.com/define-properties/-/define-properties-1.1.2.tgz#83a73f2fea569898fb737193c8f873caf6d45c94" 934 | dependencies: 935 | foreach "^2.0.5" 936 | object-keys "^1.0.8" 937 | 938 | detect-indent@^4.0.0: 939 | version "4.0.0" 940 | resolved "https://registry.yarnpkg.com/detect-indent/-/detect-indent-4.0.0.tgz#f76d064352cdf43a1cb6ce619c4ee3a9475de208" 941 | dependencies: 942 | repeating "^2.0.0" 943 | 944 | electron-to-chromium@^1.3.47: 945 | version "1.3.50" 946 | resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.3.50.tgz#7438b76f92b41b919f3fbdd350fbd0757dacddf7" 947 | 948 | error-ex@^1.2.0: 949 | version "1.3.2" 950 | resolved "https://registry.yarnpkg.com/error-ex/-/error-ex-1.3.2.tgz#b4ac40648107fdcdcfae242f428bea8a14d4f1bf" 951 | dependencies: 952 | is-arrayish "^0.2.1" 953 | 954 | error@^7.0.0: 955 | version "7.0.2" 956 | resolved "https://registry.yarnpkg.com/error/-/error-7.0.2.tgz#a5f75fff4d9926126ddac0ea5dc38e689153cb02" 957 | dependencies: 958 | string-template "~0.2.1" 959 | xtend "~4.0.0" 960 | 961 | es5-ext@^0.10.12, es5-ext@^0.10.7, es5-ext@^0.10.9, es5-ext@~0.10.11, es5-ext@~0.10.2: 962 | version "0.10.12" 963 | resolved "https://registry.yarnpkg.com/es5-ext/-/es5-ext-0.10.12.tgz#aa84641d4db76b62abba5e45fd805ecbab140047" 964 | dependencies: 965 | es6-iterator "2" 966 | es6-symbol "~3.1" 967 | 968 | es6-iterator@2: 969 | version "2.0.0" 970 | resolved "https://registry.yarnpkg.com/es6-iterator/-/es6-iterator-2.0.0.tgz#bd968567d61635e33c0b80727613c9cb4b096bac" 971 | dependencies: 972 | d "^0.1.1" 973 | es5-ext "^0.10.7" 974 | es6-symbol "3" 975 | 976 | es6-symbol@3, es6-symbol@~3.1: 977 | version "3.1.0" 978 | resolved "https://registry.yarnpkg.com/es6-symbol/-/es6-symbol-3.1.0.tgz#94481c655e7a7cad82eba832d97d5433496d7ffa" 979 | dependencies: 980 | d "~0.1.1" 981 | es5-ext "~0.10.11" 982 | 983 | es6-template-strings@^2.0.0: 984 | version "2.0.1" 985 | resolved "https://registry.yarnpkg.com/es6-template-strings/-/es6-template-strings-2.0.1.tgz#b166c6a62562f478bb7775f6ca96103a599b4b2c" 986 | dependencies: 987 | es5-ext "^0.10.12" 988 | esniff "^1.1" 989 | 990 | escape-string-regexp@^1.0.2, escape-string-regexp@^1.0.5: 991 | version "1.0.5" 992 | resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz#1b61c0562190a8dff6ae3bb2cf0200ca130b86d4" 993 | 994 | esniff@^1.1: 995 | version "1.1.0" 996 | resolved "https://registry.yarnpkg.com/esniff/-/esniff-1.1.0.tgz#c66849229f91464dede2e0d40201ed6abf65f2ac" 997 | dependencies: 998 | d "1" 999 | es5-ext "^0.10.12" 1000 | 1001 | esprima@^2.6.0: 1002 | version "2.7.3" 1003 | resolved "https://registry.yarnpkg.com/esprima/-/esprima-2.7.3.tgz#96e3b70d5779f6ad49cd032673d1c312767ba581" 1004 | 1005 | esutils@^2.0.2: 1006 | version "2.0.2" 1007 | resolved "https://registry.yarnpkg.com/esutils/-/esutils-2.0.2.tgz#0abf4f1caa5bcb1f7a9d8acc6dea4faaa04bac9b" 1008 | 1009 | eventemitter2@~0.4.13: 1010 | version "0.4.14" 1011 | resolved "https://registry.yarnpkg.com/eventemitter2/-/eventemitter2-0.4.14.tgz#8f61b75cde012b2e9eb284d4545583b5643b61ab" 1012 | 1013 | exit@~0.1.1: 1014 | version "0.1.2" 1015 | resolved "https://registry.yarnpkg.com/exit/-/exit-0.1.2.tgz#0632638f8d877cc82107d30a0fff1a17cba1cd0c" 1016 | 1017 | faye-websocket@~0.10.0: 1018 | version "0.10.0" 1019 | resolved "https://registry.yarnpkg.com/faye-websocket/-/faye-websocket-0.10.0.tgz#4e492f8d04dfb6f89003507f6edbf2d501e7c6f4" 1020 | dependencies: 1021 | websocket-driver ">=0.5.1" 1022 | 1023 | figures@^1.0.1: 1024 | version "1.7.0" 1025 | resolved "https://registry.yarnpkg.com/figures/-/figures-1.7.0.tgz#cbe1e3affcf1cd44b80cadfed28dc793a9701d2e" 1026 | dependencies: 1027 | escape-string-regexp "^1.0.5" 1028 | object-assign "^4.1.0" 1029 | 1030 | file-sync-cmp@^0.1.0: 1031 | version "0.1.1" 1032 | resolved "https://registry.yarnpkg.com/file-sync-cmp/-/file-sync-cmp-0.1.1.tgz#a5e7a8ffbfa493b43b923bbd4ca89a53b63b612b" 1033 | 1034 | find-up@^1.0.0: 1035 | version "1.1.2" 1036 | resolved "https://registry.yarnpkg.com/find-up/-/find-up-1.1.2.tgz#6b2e9822b1a2ce0a60ab64d610eccad53cb24d0f" 1037 | dependencies: 1038 | path-exists "^2.0.0" 1039 | pinkie-promise "^2.0.0" 1040 | 1041 | findup-sync@~0.3.0: 1042 | version "0.3.0" 1043 | resolved "https://registry.yarnpkg.com/findup-sync/-/findup-sync-0.3.0.tgz#37930aa5d816b777c03445e1966cc6790a4c0b16" 1044 | dependencies: 1045 | glob "~5.0.0" 1046 | 1047 | foreach@^2.0.5: 1048 | version "2.0.5" 1049 | resolved "https://registry.yarnpkg.com/foreach/-/foreach-2.0.5.tgz#0bee005018aeb260d0a3af3ae658dd0136ec1b99" 1050 | 1051 | fs.realpath@^1.0.0: 1052 | version "1.0.0" 1053 | resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f" 1054 | 1055 | function-bind@^1.1.1: 1056 | version "1.1.1" 1057 | resolved "https://registry.yarnpkg.com/function-bind/-/function-bind-1.1.1.tgz#a56899d3ea3c9bab874bb9773b7c5ede92f4895d" 1058 | 1059 | gaze@^1.1.0: 1060 | version "1.1.3" 1061 | resolved "https://registry.yarnpkg.com/gaze/-/gaze-1.1.3.tgz#c441733e13b927ac8c0ff0b4c3b033f28812924a" 1062 | dependencies: 1063 | globule "^1.0.0" 1064 | 1065 | get-stdin@^4.0.1: 1066 | version "4.0.1" 1067 | resolved "https://registry.yarnpkg.com/get-stdin/-/get-stdin-4.0.1.tgz#b968c6b0a04384324902e8bf1a5df32579a450fe" 1068 | 1069 | getobject@~0.1.0: 1070 | version "0.1.0" 1071 | resolved "https://registry.yarnpkg.com/getobject/-/getobject-0.1.0.tgz#047a449789fa160d018f5486ed91320b6ec7885c" 1072 | 1073 | glob@5.0.x, glob@~5.0.0: 1074 | version "5.0.15" 1075 | resolved "https://registry.yarnpkg.com/glob/-/glob-5.0.15.tgz#1bc936b9e02f4a603fcc222ecf7633d30b8b93b1" 1076 | dependencies: 1077 | inflight "^1.0.4" 1078 | inherits "2" 1079 | minimatch "2 || 3" 1080 | once "^1.3.0" 1081 | path-is-absolute "^1.0.0" 1082 | 1083 | glob@^7.0.3: 1084 | version "7.1.1" 1085 | resolved "https://registry.yarnpkg.com/glob/-/glob-7.1.1.tgz#805211df04faaf1c63a3600306cdf5ade50b2ec8" 1086 | dependencies: 1087 | fs.realpath "^1.0.0" 1088 | inflight "^1.0.4" 1089 | inherits "2" 1090 | minimatch "^3.0.2" 1091 | once "^1.3.0" 1092 | path-is-absolute "^1.0.0" 1093 | 1094 | glob@^7.0.5, glob@~7.1.1: 1095 | version "7.1.2" 1096 | resolved "https://registry.yarnpkg.com/glob/-/glob-7.1.2.tgz#c19c9df9a028702d678612384a6552404c636d15" 1097 | dependencies: 1098 | fs.realpath "^1.0.0" 1099 | inflight "^1.0.4" 1100 | inherits "2" 1101 | minimatch "^3.0.4" 1102 | once "^1.3.0" 1103 | path-is-absolute "^1.0.0" 1104 | 1105 | glob@~7.0.0: 1106 | version "7.0.6" 1107 | resolved "https://registry.yarnpkg.com/glob/-/glob-7.0.6.tgz#211bafaf49e525b8cd93260d14ab136152b3f57a" 1108 | dependencies: 1109 | fs.realpath "^1.0.0" 1110 | inflight "^1.0.4" 1111 | inherits "2" 1112 | minimatch "^3.0.2" 1113 | once "^1.3.0" 1114 | path-is-absolute "^1.0.0" 1115 | 1116 | globals@^9.0.0: 1117 | version "9.16.0" 1118 | resolved "https://registry.yarnpkg.com/globals/-/globals-9.16.0.tgz#63e903658171ec2d9f51b1d31de5e2b8dc01fb80" 1119 | 1120 | globals@^9.18.0: 1121 | version "9.18.0" 1122 | resolved "https://registry.yarnpkg.com/globals/-/globals-9.18.0.tgz#aa3896b3e69b487f17e31ed2143d69a8e30c2d8a" 1123 | 1124 | globule@^1.0.0: 1125 | version "1.2.1" 1126 | resolved "https://registry.yarnpkg.com/globule/-/globule-1.2.1.tgz#5dffb1b191f22d20797a9369b49eab4e9839696d" 1127 | dependencies: 1128 | glob "~7.1.1" 1129 | lodash "~4.17.10" 1130 | minimatch "~3.0.2" 1131 | 1132 | graceful-fs@^4.1.2: 1133 | version "4.1.11" 1134 | resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.1.11.tgz#0e8bdfe4d1ddb8854d64e04ea7c00e2a026e5658" 1135 | 1136 | "graceful-readlink@>= 1.0.0": 1137 | version "1.0.1" 1138 | resolved "https://registry.yarnpkg.com/graceful-readlink/-/graceful-readlink-1.0.1.tgz#4cafad76bc62f02fa039b2f94e9a3dd3a391a725" 1139 | 1140 | grunt-babel@~6.0.0: 1141 | version "6.0.0" 1142 | resolved "https://registry.yarnpkg.com/grunt-babel/-/grunt-babel-6.0.0.tgz#378189b487de1168c4c4a9fc88dd6005b35df960" 1143 | dependencies: 1144 | babel-core "^6.0.12" 1145 | 1146 | grunt-cli@^1.2.0, grunt-cli@~1.2.0: 1147 | version "1.2.0" 1148 | resolved "https://registry.yarnpkg.com/grunt-cli/-/grunt-cli-1.2.0.tgz#562b119ebb069ddb464ace2845501be97b35b6a8" 1149 | dependencies: 1150 | findup-sync "~0.3.0" 1151 | grunt-known-options "~1.1.0" 1152 | nopt "~3.0.6" 1153 | resolve "~1.1.0" 1154 | 1155 | grunt-contrib-clean@^1.1.0: 1156 | version "1.1.0" 1157 | resolved "https://registry.yarnpkg.com/grunt-contrib-clean/-/grunt-contrib-clean-1.1.0.tgz#564abf2d0378a983a15b9e3f30ee75b738c40638" 1158 | dependencies: 1159 | async "^1.5.2" 1160 | rimraf "^2.5.1" 1161 | 1162 | grunt-contrib-copy@^1.0.0: 1163 | version "1.0.0" 1164 | resolved "https://registry.yarnpkg.com/grunt-contrib-copy/-/grunt-contrib-copy-1.0.0.tgz#7060c6581e904b8ab0d00f076e0a8f6e3e7c3573" 1165 | dependencies: 1166 | chalk "^1.1.1" 1167 | file-sync-cmp "^0.1.0" 1168 | 1169 | grunt-contrib-uglify@^2.3.0: 1170 | version "2.3.0" 1171 | resolved "https://registry.yarnpkg.com/grunt-contrib-uglify/-/grunt-contrib-uglify-2.3.0.tgz#b3d0260ebdd6cefa12ff2f8e9e1e259f7de4216f" 1172 | dependencies: 1173 | chalk "^1.0.0" 1174 | maxmin "^1.1.0" 1175 | object.assign "^4.0.4" 1176 | uglify-js "~2.8.21" 1177 | uri-path "^1.0.0" 1178 | 1179 | grunt-contrib-watch@^1.0.0: 1180 | version "1.1.0" 1181 | resolved "https://registry.yarnpkg.com/grunt-contrib-watch/-/grunt-contrib-watch-1.1.0.tgz#c143ca5b824b288a024b856639a5345aedb78ed4" 1182 | dependencies: 1183 | async "^2.6.0" 1184 | gaze "^1.1.0" 1185 | lodash "^4.17.10" 1186 | tiny-lr "^1.1.1" 1187 | 1188 | grunt-execute@~0.2.2: 1189 | version "0.2.2" 1190 | resolved "https://registry.yarnpkg.com/grunt-execute/-/grunt-execute-0.2.2.tgz#4e945fe57959cc0de7799083b6b42aed9616350a" 1191 | 1192 | grunt-known-options@~1.1.0: 1193 | version "1.1.0" 1194 | resolved "https://registry.yarnpkg.com/grunt-known-options/-/grunt-known-options-1.1.0.tgz#a4274eeb32fa765da5a7a3b1712617ce3b144149" 1195 | 1196 | grunt-legacy-log-utils@~2.0.0: 1197 | version "2.0.1" 1198 | resolved "https://registry.yarnpkg.com/grunt-legacy-log-utils/-/grunt-legacy-log-utils-2.0.1.tgz#d2f442c7c0150065d9004b08fd7410d37519194e" 1199 | dependencies: 1200 | chalk "~2.4.1" 1201 | lodash "~4.17.10" 1202 | 1203 | grunt-legacy-log@~2.0.0: 1204 | version "2.0.0" 1205 | resolved "https://registry.yarnpkg.com/grunt-legacy-log/-/grunt-legacy-log-2.0.0.tgz#c8cd2c6c81a4465b9bbf2d874d963fef7a59ffb9" 1206 | dependencies: 1207 | colors "~1.1.2" 1208 | grunt-legacy-log-utils "~2.0.0" 1209 | hooker "~0.2.3" 1210 | lodash "~4.17.5" 1211 | 1212 | grunt-legacy-util@~1.1.1: 1213 | version "1.1.1" 1214 | resolved "https://registry.yarnpkg.com/grunt-legacy-util/-/grunt-legacy-util-1.1.1.tgz#e10624e7c86034e5b870c8a8616743f0a0845e42" 1215 | dependencies: 1216 | async "~1.5.2" 1217 | exit "~0.1.1" 1218 | getobject "~0.1.0" 1219 | hooker "~0.2.3" 1220 | lodash "~4.17.10" 1221 | underscore.string "~3.3.4" 1222 | which "~1.3.0" 1223 | 1224 | grunt-systemjs-builder@^1.0.0: 1225 | version "1.0.0" 1226 | resolved "https://registry.yarnpkg.com/grunt-systemjs-builder/-/grunt-systemjs-builder-1.0.0.tgz#5d8e7cbeca5b35e2b7b6bd002e9a9d7e03d7decd" 1227 | dependencies: 1228 | systemjs-builder "0.14.11 - 0.16.x" 1229 | 1230 | grunt@^1.0.1: 1231 | version "1.0.3" 1232 | resolved "https://registry.yarnpkg.com/grunt/-/grunt-1.0.3.tgz#b3c99260c51d1b42835766e796527b60f7bba374" 1233 | dependencies: 1234 | coffeescript "~1.10.0" 1235 | dateformat "~1.0.12" 1236 | eventemitter2 "~0.4.13" 1237 | exit "~0.1.1" 1238 | findup-sync "~0.3.0" 1239 | glob "~7.0.0" 1240 | grunt-cli "~1.2.0" 1241 | grunt-known-options "~1.1.0" 1242 | grunt-legacy-log "~2.0.0" 1243 | grunt-legacy-util "~1.1.1" 1244 | iconv-lite "~0.4.13" 1245 | js-yaml "~3.5.2" 1246 | minimatch "~3.0.2" 1247 | mkdirp "~0.5.1" 1248 | nopt "~3.0.6" 1249 | path-is-absolute "~1.0.0" 1250 | rimraf "~2.6.2" 1251 | 1252 | gzip-size@^1.0.0: 1253 | version "1.0.0" 1254 | resolved "https://registry.yarnpkg.com/gzip-size/-/gzip-size-1.0.0.tgz#66cf8b101047227b95bace6ea1da0c177ed5c22f" 1255 | dependencies: 1256 | browserify-zlib "^0.1.4" 1257 | concat-stream "^1.4.1" 1258 | 1259 | has-ansi@^2.0.0: 1260 | version "2.0.0" 1261 | resolved "https://registry.yarnpkg.com/has-ansi/-/has-ansi-2.0.0.tgz#34f5049ce1ecdf2b0649af3ef24e45ed35416d91" 1262 | dependencies: 1263 | ansi-regex "^2.0.0" 1264 | 1265 | has-flag@^3.0.0: 1266 | version "3.0.0" 1267 | resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-3.0.0.tgz#b5d454dc2199ae225699f3467e5a07f3b955bafd" 1268 | 1269 | has-symbols@^1.0.0: 1270 | version "1.0.0" 1271 | resolved "https://registry.yarnpkg.com/has-symbols/-/has-symbols-1.0.0.tgz#ba1a8f1af2a0fc39650f5c850367704122063b44" 1272 | 1273 | home-or-tmp@^2.0.0: 1274 | version "2.0.0" 1275 | resolved "https://registry.yarnpkg.com/home-or-tmp/-/home-or-tmp-2.0.0.tgz#e36c3f2d2cae7d746a857e38d18d5f32a7882db8" 1276 | dependencies: 1277 | os-homedir "^1.0.0" 1278 | os-tmpdir "^1.0.1" 1279 | 1280 | hooker@~0.2.3: 1281 | version "0.2.3" 1282 | resolved "https://registry.yarnpkg.com/hooker/-/hooker-0.2.3.tgz#b834f723cc4a242aa65963459df6d984c5d3d959" 1283 | 1284 | hosted-git-info@^2.1.4: 1285 | version "2.6.1" 1286 | resolved "https://registry.yarnpkg.com/hosted-git-info/-/hosted-git-info-2.6.1.tgz#6e4cee78b01bb849dcf93527708c69fdbee410df" 1287 | 1288 | http-parser-js@>=0.4.0: 1289 | version "0.4.13" 1290 | resolved "https://registry.yarnpkg.com/http-parser-js/-/http-parser-js-0.4.13.tgz#3bd6d6fde6e3172c9334c3b33b6c193d80fe1137" 1291 | 1292 | iconv-lite@~0.4.13: 1293 | version "0.4.23" 1294 | resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.23.tgz#297871f63be507adcfbfca715d0cd0eed84e9a63" 1295 | dependencies: 1296 | safer-buffer ">= 2.1.2 < 3" 1297 | 1298 | indent-string@^2.1.0: 1299 | version "2.1.0" 1300 | resolved "https://registry.yarnpkg.com/indent-string/-/indent-string-2.1.0.tgz#8e2d48348742121b4a8218b7a137e9a52049dc80" 1301 | dependencies: 1302 | repeating "^2.0.0" 1303 | 1304 | inflight@^1.0.4: 1305 | version "1.0.6" 1306 | resolved "https://registry.yarnpkg.com/inflight/-/inflight-1.0.6.tgz#49bd6331d7d02d0c09bc910a1075ba8165b56df9" 1307 | dependencies: 1308 | once "^1.3.0" 1309 | wrappy "1" 1310 | 1311 | inherits@2, inherits@^2.0.3, inherits@~2.0.3: 1312 | version "2.0.3" 1313 | resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.3.tgz#633c2c83e3da42a502f52466022480f4208261de" 1314 | 1315 | invariant@^2.2.0: 1316 | version "2.2.2" 1317 | resolved "https://registry.yarnpkg.com/invariant/-/invariant-2.2.2.tgz#9e1f56ac0acdb6bf303306f338be3b204ae60360" 1318 | dependencies: 1319 | loose-envify "^1.0.0" 1320 | 1321 | invariant@^2.2.2: 1322 | version "2.2.4" 1323 | resolved "https://registry.yarnpkg.com/invariant/-/invariant-2.2.4.tgz#610f3c92c9359ce1db616e538008d23ff35158e6" 1324 | dependencies: 1325 | loose-envify "^1.0.0" 1326 | 1327 | is-arrayish@^0.2.1: 1328 | version "0.2.1" 1329 | resolved "https://registry.yarnpkg.com/is-arrayish/-/is-arrayish-0.2.1.tgz#77c99840527aa8ecb1a8ba697b80645a7a926a9d" 1330 | 1331 | is-buffer@^1.0.2: 1332 | version "1.1.4" 1333 | resolved "https://registry.yarnpkg.com/is-buffer/-/is-buffer-1.1.4.tgz#cfc86ccd5dc5a52fa80489111c6920c457e2d98b" 1334 | 1335 | is-builtin-module@^1.0.0: 1336 | version "1.0.0" 1337 | resolved "https://registry.yarnpkg.com/is-builtin-module/-/is-builtin-module-1.0.0.tgz#540572d34f7ac3119f8f76c30cbc1b1e037affbe" 1338 | dependencies: 1339 | builtin-modules "^1.0.0" 1340 | 1341 | is-finite@^1.0.0: 1342 | version "1.0.2" 1343 | resolved "https://registry.yarnpkg.com/is-finite/-/is-finite-1.0.2.tgz#cc6677695602be550ef11e8b4aa6305342b6d0aa" 1344 | dependencies: 1345 | number-is-nan "^1.0.0" 1346 | 1347 | is-utf8@^0.2.0: 1348 | version "0.2.1" 1349 | resolved "https://registry.yarnpkg.com/is-utf8/-/is-utf8-0.2.1.tgz#4b0da1442104d1b336340e80797e865cf39f7d72" 1350 | 1351 | isarray@~1.0.0: 1352 | version "1.0.0" 1353 | resolved "https://registry.yarnpkg.com/isarray/-/isarray-1.0.0.tgz#bb935d48582cba168c06834957a54a3e07124f11" 1354 | 1355 | isexe@^2.0.0: 1356 | version "2.0.0" 1357 | resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10" 1358 | 1359 | js-tokens@^3.0.0: 1360 | version "3.0.1" 1361 | resolved "https://registry.yarnpkg.com/js-tokens/-/js-tokens-3.0.1.tgz#08e9f132484a2c45a30907e9dc4d5567b7f114d7" 1362 | 1363 | js-tokens@^3.0.2: 1364 | version "3.0.2" 1365 | resolved "https://registry.yarnpkg.com/js-tokens/-/js-tokens-3.0.2.tgz#9866df395102130e38f7f996bceb65443209c25b" 1366 | 1367 | js-yaml@~3.5.2: 1368 | version "3.5.5" 1369 | resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-3.5.5.tgz#0377c38017cabc7322b0d1fbcd25a491641f2fbe" 1370 | dependencies: 1371 | argparse "^1.0.2" 1372 | esprima "^2.6.0" 1373 | 1374 | jsesc@^1.3.0: 1375 | version "1.3.0" 1376 | resolved "https://registry.yarnpkg.com/jsesc/-/jsesc-1.3.0.tgz#46c3fec8c1892b12b0833db9bc7622176dbab34b" 1377 | 1378 | jsesc@~0.5.0: 1379 | version "0.5.0" 1380 | resolved "https://registry.yarnpkg.com/jsesc/-/jsesc-0.5.0.tgz#e7dee66e35d6fc16f710fe91d5cf69f70f08911d" 1381 | 1382 | json5@^0.5.0, json5@^0.5.1: 1383 | version "0.5.1" 1384 | resolved "https://registry.yarnpkg.com/json5/-/json5-0.5.1.tgz#1eade7acc012034ad84e2396767ead9fa5495821" 1385 | 1386 | kind-of@^3.0.2: 1387 | version "3.1.0" 1388 | resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-3.1.0.tgz#475d698a5e49ff5e53d14e3e732429dc8bf4cf47" 1389 | dependencies: 1390 | is-buffer "^1.0.2" 1391 | 1392 | lazy-cache@^1.0.3: 1393 | version "1.0.4" 1394 | resolved "https://registry.yarnpkg.com/lazy-cache/-/lazy-cache-1.0.4.tgz#a1d78fc3a50474cb80845d3b3b6e1da49a446e8e" 1395 | 1396 | livereload-js@^2.3.0: 1397 | version "2.3.0" 1398 | resolved "https://registry.yarnpkg.com/livereload-js/-/livereload-js-2.3.0.tgz#c3ab22e8aaf5bf3505d80d098cbad67726548c9a" 1399 | 1400 | load-grunt-tasks@^3.5.2: 1401 | version "3.5.2" 1402 | resolved "https://registry.yarnpkg.com/load-grunt-tasks/-/load-grunt-tasks-3.5.2.tgz#0728561180fd20ff8a6927505852fc58aaea0c88" 1403 | dependencies: 1404 | arrify "^1.0.0" 1405 | multimatch "^2.0.0" 1406 | pkg-up "^1.0.0" 1407 | resolve-pkg "^0.1.0" 1408 | 1409 | load-json-file@^1.0.0: 1410 | version "1.1.0" 1411 | resolved "https://registry.yarnpkg.com/load-json-file/-/load-json-file-1.1.0.tgz#956905708d58b4bab4c2261b04f59f31c99374c0" 1412 | dependencies: 1413 | graceful-fs "^4.1.2" 1414 | parse-json "^2.2.0" 1415 | pify "^2.0.0" 1416 | pinkie-promise "^2.0.0" 1417 | strip-bom "^2.0.0" 1418 | 1419 | lodash@^4.17.10, lodash@^4.17.4, lodash@~4.17.10, lodash@~4.17.5: 1420 | version "4.17.10" 1421 | resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.10.tgz#1b7793cf7259ea38fb3661d4d38b3260af8ae4e7" 1422 | 1423 | lodash@^4.2.0: 1424 | version "4.17.4" 1425 | resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.4.tgz#78203a4d1c328ae1d86dca6460e369b57f4055ae" 1426 | 1427 | longest@^1.0.1: 1428 | version "1.0.1" 1429 | resolved "https://registry.yarnpkg.com/longest/-/longest-1.0.1.tgz#30a0b2da38f73770e8294a0d22e6625ed77d0097" 1430 | 1431 | loose-envify@^1.0.0: 1432 | version "1.3.1" 1433 | resolved "https://registry.yarnpkg.com/loose-envify/-/loose-envify-1.3.1.tgz#d1a8ad33fa9ce0e713d65fdd0ac8b748d478c848" 1434 | dependencies: 1435 | js-tokens "^3.0.0" 1436 | 1437 | loud-rejection@^1.0.0: 1438 | version "1.6.0" 1439 | resolved "https://registry.yarnpkg.com/loud-rejection/-/loud-rejection-1.6.0.tgz#5b46f80147edee578870f086d04821cf998e551f" 1440 | dependencies: 1441 | currently-unhandled "^0.4.1" 1442 | signal-exit "^3.0.0" 1443 | 1444 | map-obj@^1.0.0, map-obj@^1.0.1: 1445 | version "1.0.1" 1446 | resolved "https://registry.yarnpkg.com/map-obj/-/map-obj-1.0.1.tgz#d933ceb9205d82bdcf4886f6742bdc2b4dea146d" 1447 | 1448 | maxmin@^1.1.0: 1449 | version "1.1.0" 1450 | resolved "https://registry.yarnpkg.com/maxmin/-/maxmin-1.1.0.tgz#71365e84a99dd8f8b3f7d5fde2f00d1e7f73be61" 1451 | dependencies: 1452 | chalk "^1.0.0" 1453 | figures "^1.0.1" 1454 | gzip-size "^1.0.0" 1455 | pretty-bytes "^1.0.0" 1456 | 1457 | meow@^3.1.0, meow@^3.3.0: 1458 | version "3.7.0" 1459 | resolved "https://registry.yarnpkg.com/meow/-/meow-3.7.0.tgz#72cb668b425228290abbfa856892587308a801fb" 1460 | dependencies: 1461 | camelcase-keys "^2.0.0" 1462 | decamelize "^1.1.2" 1463 | loud-rejection "^1.0.0" 1464 | map-obj "^1.0.1" 1465 | minimist "^1.1.3" 1466 | normalize-package-data "^2.3.4" 1467 | object-assign "^4.0.1" 1468 | read-pkg-up "^1.0.1" 1469 | redent "^1.0.0" 1470 | trim-newlines "^1.0.0" 1471 | 1472 | "minimatch@2 || 3", minimatch@^3.0.0, minimatch@^3.0.2: 1473 | version "3.0.3" 1474 | resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.0.3.tgz#2a4e4090b96b2db06a9d7df01055a62a77c9b774" 1475 | dependencies: 1476 | brace-expansion "^1.0.0" 1477 | 1478 | minimatch@^3.0.4, minimatch@~3.0.2: 1479 | version "3.0.4" 1480 | resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.0.4.tgz#5166e286457f03306064be5497e8dbb0c3d32083" 1481 | dependencies: 1482 | brace-expansion "^1.1.7" 1483 | 1484 | minimist@0.0.8: 1485 | version "0.0.8" 1486 | resolved "https://registry.yarnpkg.com/minimist/-/minimist-0.0.8.tgz#857fcabfc3397d2625b8228262e86aa7a011b05d" 1487 | 1488 | minimist@^1.1.3: 1489 | version "1.2.0" 1490 | resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.0.tgz#a35008b20f41383eec1fb914f4cd5df79a264284" 1491 | 1492 | mkdirp@^0.5.1, mkdirp@~0.5.1: 1493 | version "0.5.1" 1494 | resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-0.5.1.tgz#30057438eac6cf7f8c4767f38648d6697d75c903" 1495 | dependencies: 1496 | minimist "0.0.8" 1497 | 1498 | moment@^2.24.0: 1499 | version "2.24.0" 1500 | resolved "https://registry.yarnpkg.com/moment/-/moment-2.24.0.tgz#0d055d53f5052aa653c9f6eb68bb5d12bf5c2b5b" 1501 | integrity sha512-bV7f+6l2QigeBBZSM/6yTNq4P2fNpSWj/0e7jQcy87A8e7o2nAfP/34/2ky5Vw4B9S446EtIhodAzkFCcR4dQg== 1502 | 1503 | ms@0.7.1: 1504 | version "0.7.1" 1505 | resolved "https://registry.yarnpkg.com/ms/-/ms-0.7.1.tgz#9cd13c03adbff25b65effde7ce864ee952017098" 1506 | 1507 | ms@2.0.0: 1508 | version "2.0.0" 1509 | resolved "https://registry.yarnpkg.com/ms/-/ms-2.0.0.tgz#5608aeadfc00be6c2901df5f9861788de0d597c8" 1510 | 1511 | multimatch@^2.0.0: 1512 | version "2.1.0" 1513 | resolved "https://registry.yarnpkg.com/multimatch/-/multimatch-2.1.0.tgz#9c7906a22fb4c02919e2f5f75161b4cdbd4b2a2b" 1514 | dependencies: 1515 | array-differ "^1.0.0" 1516 | array-union "^1.0.1" 1517 | arrify "^1.0.0" 1518 | minimatch "^3.0.0" 1519 | 1520 | nopt@~3.0.6: 1521 | version "3.0.6" 1522 | resolved "https://registry.yarnpkg.com/nopt/-/nopt-3.0.6.tgz#c6465dbf08abcd4db359317f79ac68a646b28ff9" 1523 | dependencies: 1524 | abbrev "1" 1525 | 1526 | normalize-package-data@^2.3.2, normalize-package-data@^2.3.4: 1527 | version "2.4.0" 1528 | resolved "https://registry.yarnpkg.com/normalize-package-data/-/normalize-package-data-2.4.0.tgz#12f95a307d58352075a04907b84ac8be98ac012f" 1529 | dependencies: 1530 | hosted-git-info "^2.1.4" 1531 | is-builtin-module "^1.0.0" 1532 | semver "2 || 3 || 4 || 5" 1533 | validate-npm-package-license "^3.0.1" 1534 | 1535 | number-is-nan@^1.0.0: 1536 | version "1.0.1" 1537 | resolved "https://registry.yarnpkg.com/number-is-nan/-/number-is-nan-1.0.1.tgz#097b602b53422a522c1afb8790318336941a011d" 1538 | 1539 | object-assign@^4.0.1, object-assign@^4.1.0: 1540 | version "4.1.1" 1541 | resolved "https://registry.yarnpkg.com/object-assign/-/object-assign-4.1.1.tgz#2109adc7965887cfc05cbbd442cac8bfbb360863" 1542 | 1543 | object-keys@^1.0.11, object-keys@^1.0.8: 1544 | version "1.0.12" 1545 | resolved "https://registry.yarnpkg.com/object-keys/-/object-keys-1.0.12.tgz#09c53855377575310cca62f55bb334abff7b3ed2" 1546 | 1547 | object.assign@^4.0.4: 1548 | version "4.1.0" 1549 | resolved "https://registry.yarnpkg.com/object.assign/-/object.assign-4.1.0.tgz#968bf1100d7956bb3ca086f006f846b3bc4008da" 1550 | dependencies: 1551 | define-properties "^1.1.2" 1552 | function-bind "^1.1.1" 1553 | has-symbols "^1.0.0" 1554 | object-keys "^1.0.11" 1555 | 1556 | once@^1.3.0: 1557 | version "1.4.0" 1558 | resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1" 1559 | dependencies: 1560 | wrappy "1" 1561 | 1562 | os-homedir@^1.0.0: 1563 | version "1.0.2" 1564 | resolved "https://registry.yarnpkg.com/os-homedir/-/os-homedir-1.0.2.tgz#ffbc4988336e0e833de0c168c7ef152121aa7fb3" 1565 | 1566 | os-tmpdir@^1.0.1: 1567 | version "1.0.2" 1568 | resolved "https://registry.yarnpkg.com/os-tmpdir/-/os-tmpdir-1.0.2.tgz#bbe67406c79aa85c5cfec766fe5734555dfa1274" 1569 | 1570 | pako@~0.2.0: 1571 | version "0.2.9" 1572 | resolved "https://registry.yarnpkg.com/pako/-/pako-0.2.9.tgz#f3f7522f4ef782348da8161bad9ecfd51bf83a75" 1573 | 1574 | parse-json@^2.2.0: 1575 | version "2.2.0" 1576 | resolved "https://registry.yarnpkg.com/parse-json/-/parse-json-2.2.0.tgz#f480f40434ef80741f8469099f8dea18f55a4dc9" 1577 | dependencies: 1578 | error-ex "^1.2.0" 1579 | 1580 | path-exists@^2.0.0: 1581 | version "2.1.0" 1582 | resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-2.1.0.tgz#0feb6c64f0fc518d9a754dd5efb62c7022761f4b" 1583 | dependencies: 1584 | pinkie-promise "^2.0.0" 1585 | 1586 | path-is-absolute@^1.0.0, path-is-absolute@^1.0.1, path-is-absolute@~1.0.0: 1587 | version "1.0.1" 1588 | resolved "https://registry.yarnpkg.com/path-is-absolute/-/path-is-absolute-1.0.1.tgz#174b9268735534ffbc7ace6bf53a5a9e1b5c5f5f" 1589 | 1590 | path-type@^1.0.0: 1591 | version "1.1.0" 1592 | resolved "https://registry.yarnpkg.com/path-type/-/path-type-1.1.0.tgz#59c44f7ee491da704da415da5a4070ba4f8fe441" 1593 | dependencies: 1594 | graceful-fs "^4.1.2" 1595 | pify "^2.0.0" 1596 | pinkie-promise "^2.0.0" 1597 | 1598 | pify@^2.0.0: 1599 | version "2.3.0" 1600 | resolved "https://registry.yarnpkg.com/pify/-/pify-2.3.0.tgz#ed141a6ac043a849ea588498e7dca8b15330e90c" 1601 | 1602 | pinkie-promise@^2.0.0: 1603 | version "2.0.1" 1604 | resolved "https://registry.yarnpkg.com/pinkie-promise/-/pinkie-promise-2.0.1.tgz#2135d6dfa7a358c069ac9b178776288228450ffa" 1605 | dependencies: 1606 | pinkie "^2.0.0" 1607 | 1608 | pinkie@^2.0.0: 1609 | version "2.0.4" 1610 | resolved "https://registry.yarnpkg.com/pinkie/-/pinkie-2.0.4.tgz#72556b80cfa0d48a974e80e77248e80ed4f7f870" 1611 | 1612 | pkg-up@^1.0.0: 1613 | version "1.0.0" 1614 | resolved "https://registry.yarnpkg.com/pkg-up/-/pkg-up-1.0.0.tgz#3e08fb461525c4421624a33b9f7e6d0af5b05a26" 1615 | dependencies: 1616 | find-up "^1.0.0" 1617 | 1618 | pretty-bytes@^1.0.0: 1619 | version "1.0.4" 1620 | resolved "https://registry.yarnpkg.com/pretty-bytes/-/pretty-bytes-1.0.4.tgz#0a22e8210609ad35542f8c8d5d2159aff0751c84" 1621 | dependencies: 1622 | get-stdin "^4.0.1" 1623 | meow "^3.1.0" 1624 | 1625 | private@^0.1.6: 1626 | version "0.1.7" 1627 | resolved "https://registry.yarnpkg.com/private/-/private-0.1.7.tgz#68ce5e8a1ef0a23bb570cc28537b5332aba63ef1" 1628 | 1629 | private@^0.1.8: 1630 | version "0.1.8" 1631 | resolved "https://registry.yarnpkg.com/private/-/private-0.1.8.tgz#2381edb3689f7a53d653190060fcf822d2f368ff" 1632 | 1633 | process-nextick-args@~2.0.0: 1634 | version "2.0.0" 1635 | resolved "https://registry.yarnpkg.com/process-nextick-args/-/process-nextick-args-2.0.0.tgz#a37d732f4271b4ab1ad070d35508e8290788ffaa" 1636 | 1637 | qs@^6.4.0: 1638 | version "6.5.2" 1639 | resolved "https://registry.yarnpkg.com/qs/-/qs-6.5.2.tgz#cb3ae806e8740444584ef154ce8ee98d403f3e36" 1640 | 1641 | raw-body@~1.1.0: 1642 | version "1.1.7" 1643 | resolved "https://registry.yarnpkg.com/raw-body/-/raw-body-1.1.7.tgz#1d027c2bfa116acc6623bca8f00016572a87d425" 1644 | dependencies: 1645 | bytes "1" 1646 | string_decoder "0.10" 1647 | 1648 | read-pkg-up@^1.0.1: 1649 | version "1.0.1" 1650 | resolved "https://registry.yarnpkg.com/read-pkg-up/-/read-pkg-up-1.0.1.tgz#9d63c13276c065918d57f002a57f40a1b643fb02" 1651 | dependencies: 1652 | find-up "^1.0.0" 1653 | read-pkg "^1.0.0" 1654 | 1655 | read-pkg@^1.0.0: 1656 | version "1.1.0" 1657 | resolved "https://registry.yarnpkg.com/read-pkg/-/read-pkg-1.1.0.tgz#f5ffaa5ecd29cb31c0474bca7d756b6bb29e3f28" 1658 | dependencies: 1659 | load-json-file "^1.0.0" 1660 | normalize-package-data "^2.3.2" 1661 | path-type "^1.0.0" 1662 | 1663 | readable-stream@^2.2.2: 1664 | version "2.3.6" 1665 | resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-2.3.6.tgz#b11c27d88b8ff1fbe070643cf94b0c79ae1b0aaf" 1666 | dependencies: 1667 | core-util-is "~1.0.0" 1668 | inherits "~2.0.3" 1669 | isarray "~1.0.0" 1670 | process-nextick-args "~2.0.0" 1671 | safe-buffer "~5.1.1" 1672 | string_decoder "~1.1.1" 1673 | util-deprecate "~1.0.1" 1674 | 1675 | redent@^1.0.0: 1676 | version "1.0.0" 1677 | resolved "https://registry.yarnpkg.com/redent/-/redent-1.0.0.tgz#cf916ab1fd5f1f16dfb20822dd6ec7f730c2afde" 1678 | dependencies: 1679 | indent-string "^2.1.0" 1680 | strip-indent "^1.0.1" 1681 | 1682 | regenerate@^1.2.1: 1683 | version "1.3.2" 1684 | resolved "https://registry.yarnpkg.com/regenerate/-/regenerate-1.3.2.tgz#d1941c67bad437e1be76433add5b385f95b19260" 1685 | 1686 | regenerator-runtime@^0.10.0: 1687 | version "0.10.3" 1688 | resolved "https://registry.yarnpkg.com/regenerator-runtime/-/regenerator-runtime-0.10.3.tgz#8c4367a904b51ea62a908ac310bf99ff90a82a3e" 1689 | 1690 | regenerator-runtime@^0.11.0: 1691 | version "0.11.1" 1692 | resolved "https://registry.yarnpkg.com/regenerator-runtime/-/regenerator-runtime-0.11.1.tgz#be05ad7f9bf7d22e056f9726cee5017fbf19e2e9" 1693 | 1694 | regenerator-transform@^0.10.0: 1695 | version "0.10.1" 1696 | resolved "https://registry.yarnpkg.com/regenerator-transform/-/regenerator-transform-0.10.1.tgz#1e4996837231da8b7f3cf4114d71b5691a0680dd" 1697 | dependencies: 1698 | babel-runtime "^6.18.0" 1699 | babel-types "^6.19.0" 1700 | private "^0.1.6" 1701 | 1702 | regexpu-core@^2.0.0: 1703 | version "2.0.0" 1704 | resolved "https://registry.yarnpkg.com/regexpu-core/-/regexpu-core-2.0.0.tgz#49d038837b8dcf8bfa5b9a42139938e6ea2ae240" 1705 | dependencies: 1706 | regenerate "^1.2.1" 1707 | regjsgen "^0.2.0" 1708 | regjsparser "^0.1.4" 1709 | 1710 | regjsgen@^0.2.0: 1711 | version "0.2.0" 1712 | resolved "https://registry.yarnpkg.com/regjsgen/-/regjsgen-0.2.0.tgz#6c016adeac554f75823fe37ac05b92d5a4edb1f7" 1713 | 1714 | regjsparser@^0.1.4: 1715 | version "0.1.5" 1716 | resolved "https://registry.yarnpkg.com/regjsparser/-/regjsparser-0.1.5.tgz#7ee8f84dc6fa792d3fd0ae228d24bd949ead205c" 1717 | dependencies: 1718 | jsesc "~0.5.0" 1719 | 1720 | repeat-string@^1.5.2: 1721 | version "1.6.1" 1722 | resolved "https://registry.yarnpkg.com/repeat-string/-/repeat-string-1.6.1.tgz#8dcae470e1c88abc2d600fff4a776286da75e637" 1723 | 1724 | repeating@^2.0.0: 1725 | version "2.0.1" 1726 | resolved "https://registry.yarnpkg.com/repeating/-/repeating-2.0.1.tgz#5214c53a926d3552707527fbab415dbc08d06dda" 1727 | dependencies: 1728 | is-finite "^1.0.0" 1729 | 1730 | resolve-from@^2.0.0: 1731 | version "2.0.0" 1732 | resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-2.0.0.tgz#9480ab20e94ffa1d9e80a804c7ea147611966b57" 1733 | 1734 | resolve-pkg@^0.1.0: 1735 | version "0.1.0" 1736 | resolved "https://registry.yarnpkg.com/resolve-pkg/-/resolve-pkg-0.1.0.tgz#02cc993410e2936962bd97166a1b077da9725531" 1737 | dependencies: 1738 | resolve-from "^2.0.0" 1739 | 1740 | resolve@~1.1.0: 1741 | version "1.1.7" 1742 | resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.1.7.tgz#203114d82ad2c5ed9e8e0411b3932875e889e97b" 1743 | 1744 | right-align@^0.1.1: 1745 | version "0.1.3" 1746 | resolved "https://registry.yarnpkg.com/right-align/-/right-align-0.1.3.tgz#61339b722fe6a3515689210d24e14c96148613ef" 1747 | dependencies: 1748 | align-text "^0.1.1" 1749 | 1750 | rimraf@^2.5.1, rimraf@~2.6.2: 1751 | version "2.6.2" 1752 | resolved "https://registry.yarnpkg.com/rimraf/-/rimraf-2.6.2.tgz#2ed8150d24a16ea8651e6d6ef0f47c4158ce7a36" 1753 | dependencies: 1754 | glob "^7.0.5" 1755 | 1756 | rollup@^0.58.2: 1757 | version "0.58.2" 1758 | resolved "https://registry.yarnpkg.com/rollup/-/rollup-0.58.2.tgz#2feddea8c0c022f3e74b35c48e3c21b3433803ce" 1759 | dependencies: 1760 | "@types/estree" "0.0.38" 1761 | "@types/node" "*" 1762 | 1763 | rsvp@^3.0.13: 1764 | version "3.3.3" 1765 | resolved "https://registry.yarnpkg.com/rsvp/-/rsvp-3.3.3.tgz#34633caaf8bc66ceff4be3c2e1dffd032538a813" 1766 | 1767 | safe-buffer@~5.1.0, safe-buffer@~5.1.1: 1768 | version "5.1.2" 1769 | resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.1.2.tgz#991ec69d296e0313747d59bdfd2b745c35f8828d" 1770 | 1771 | safe-json-parse@~1.0.1: 1772 | version "1.0.1" 1773 | resolved "https://registry.yarnpkg.com/safe-json-parse/-/safe-json-parse-1.0.1.tgz#3e76723e38dfdda13c9b1d29a1e07ffee4b30b57" 1774 | 1775 | "safer-buffer@>= 2.1.2 < 3": 1776 | version "2.1.2" 1777 | resolved "https://registry.yarnpkg.com/safer-buffer/-/safer-buffer-2.1.2.tgz#44fa161b0187b9549dd84bb91802f9bd8385cd6a" 1778 | 1779 | "semver@2 || 3 || 4 || 5", semver@^5.3.0: 1780 | version "5.5.0" 1781 | resolved "https://registry.yarnpkg.com/semver/-/semver-5.5.0.tgz#dc4bbc7a6ca9d916dee5d43516f0092b58f7b8ab" 1782 | 1783 | semver@^4.3.3: 1784 | version "4.3.6" 1785 | resolved "https://registry.yarnpkg.com/semver/-/semver-4.3.6.tgz#300bc6e0e86374f7ba61068b5b1ecd57fc6532da" 1786 | 1787 | signal-exit@^3.0.0: 1788 | version "3.0.2" 1789 | resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-3.0.2.tgz#b5fdc08f1287ea1178628e415e25132b73646c6d" 1790 | 1791 | slash@^1.0.0: 1792 | version "1.0.0" 1793 | resolved "https://registry.yarnpkg.com/slash/-/slash-1.0.0.tgz#c41f2f6c39fc16d1cd17ad4b5d896114ae470d55" 1794 | 1795 | source-map-support@^0.4.15: 1796 | version "0.4.18" 1797 | resolved "https://registry.yarnpkg.com/source-map-support/-/source-map-support-0.4.18.tgz#0286a6de8be42641338594e97ccea75f0a2c585f" 1798 | dependencies: 1799 | source-map "^0.5.6" 1800 | 1801 | source-map-support@^0.4.2: 1802 | version "0.4.11" 1803 | resolved "https://registry.yarnpkg.com/source-map-support/-/source-map-support-0.4.11.tgz#647f939978b38535909530885303daf23279f322" 1804 | dependencies: 1805 | source-map "^0.5.3" 1806 | 1807 | source-map-support@~0.2.8: 1808 | version "0.2.10" 1809 | resolved "https://registry.yarnpkg.com/source-map-support/-/source-map-support-0.2.10.tgz#ea5a3900a1c1cb25096a0ae8cc5c2b4b10ded3dc" 1810 | dependencies: 1811 | source-map "0.1.32" 1812 | 1813 | source-map@0.1.32: 1814 | version "0.1.32" 1815 | resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.1.32.tgz#c8b6c167797ba4740a8ea33252162ff08591b266" 1816 | dependencies: 1817 | amdefine ">=0.0.4" 1818 | 1819 | source-map@^0.5.0, source-map@^0.5.3, source-map@~0.5.1: 1820 | version "0.5.6" 1821 | resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.5.6.tgz#75ce38f52bf0733c5a7f0c118d81334a2bb5f412" 1822 | 1823 | source-map@^0.5.6, source-map@^0.5.7: 1824 | version "0.5.7" 1825 | resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.5.7.tgz#8a039d2d1021d22d1ea14c80d8ea468ba2ef3fcc" 1826 | 1827 | spdx-correct@^3.0.0: 1828 | version "3.0.0" 1829 | resolved "https://registry.yarnpkg.com/spdx-correct/-/spdx-correct-3.0.0.tgz#05a5b4d7153a195bc92c3c425b69f3b2a9524c82" 1830 | dependencies: 1831 | spdx-expression-parse "^3.0.0" 1832 | spdx-license-ids "^3.0.0" 1833 | 1834 | spdx-exceptions@^2.1.0: 1835 | version "2.1.0" 1836 | resolved "https://registry.yarnpkg.com/spdx-exceptions/-/spdx-exceptions-2.1.0.tgz#2c7ae61056c714a5b9b9b2b2af7d311ef5c78fe9" 1837 | 1838 | spdx-expression-parse@^3.0.0: 1839 | version "3.0.0" 1840 | resolved "https://registry.yarnpkg.com/spdx-expression-parse/-/spdx-expression-parse-3.0.0.tgz#99e119b7a5da00e05491c9fa338b7904823b41d0" 1841 | dependencies: 1842 | spdx-exceptions "^2.1.0" 1843 | spdx-license-ids "^3.0.0" 1844 | 1845 | spdx-license-ids@^3.0.0: 1846 | version "3.0.0" 1847 | resolved "https://registry.yarnpkg.com/spdx-license-ids/-/spdx-license-ids-3.0.0.tgz#7a7cd28470cc6d3a1cfe6d66886f6bc430d3ac87" 1848 | 1849 | sprintf-js@^1.0.3: 1850 | version "1.1.1" 1851 | resolved "https://registry.yarnpkg.com/sprintf-js/-/sprintf-js-1.1.1.tgz#36be78320afe5801f6cea3ee78b6e5aab940ea0c" 1852 | 1853 | sprintf-js@~1.0.2: 1854 | version "1.0.3" 1855 | resolved "https://registry.yarnpkg.com/sprintf-js/-/sprintf-js-1.0.3.tgz#04e6926f662895354f3dd015203633b857297e2c" 1856 | 1857 | string-template@~0.2.1: 1858 | version "0.2.1" 1859 | resolved "https://registry.yarnpkg.com/string-template/-/string-template-0.2.1.tgz#42932e598a352d01fc22ec3367d9d84eec6c9add" 1860 | 1861 | string_decoder@0.10: 1862 | version "0.10.31" 1863 | resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-0.10.31.tgz#62e203bc41766c6c28c9fc84301dab1c5310fa94" 1864 | 1865 | string_decoder@~1.1.1: 1866 | version "1.1.1" 1867 | resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.1.1.tgz#9cf1611ba62685d7030ae9e4ba34149c3af03fc8" 1868 | dependencies: 1869 | safe-buffer "~5.1.0" 1870 | 1871 | strip-ansi@^3.0.0: 1872 | version "3.0.1" 1873 | resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-3.0.1.tgz#6a385fb8853d952d5ff05d0e8aaf94278dc63dcf" 1874 | dependencies: 1875 | ansi-regex "^2.0.0" 1876 | 1877 | strip-bom@^2.0.0: 1878 | version "2.0.0" 1879 | resolved "https://registry.yarnpkg.com/strip-bom/-/strip-bom-2.0.0.tgz#6219a85616520491f35788bdbf1447a99c7e6b0e" 1880 | dependencies: 1881 | is-utf8 "^0.2.0" 1882 | 1883 | strip-indent@^1.0.1: 1884 | version "1.0.1" 1885 | resolved "https://registry.yarnpkg.com/strip-indent/-/strip-indent-1.0.1.tgz#0c7962a6adefa7bbd4ac366460a638552ae1a0a2" 1886 | dependencies: 1887 | get-stdin "^4.0.1" 1888 | 1889 | supports-color@^2.0.0: 1890 | version "2.0.0" 1891 | resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-2.0.0.tgz#535d045ce6b6363fa40117084629995e9df324c7" 1892 | 1893 | supports-color@^5.3.0: 1894 | version "5.4.0" 1895 | resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-5.4.0.tgz#1c6b337402c2137605efe19f10fec390f6faab54" 1896 | dependencies: 1897 | has-flag "^3.0.0" 1898 | 1899 | "systemjs-builder@0.14.11 - 0.16.x": 1900 | version "0.16.13" 1901 | resolved "https://registry.yarnpkg.com/systemjs-builder/-/systemjs-builder-0.16.13.tgz#02b47d03afd1e2f29562b11ec8bc13457e785c76" 1902 | dependencies: 1903 | babel-core "^6.24.1" 1904 | babel-plugin-syntax-dynamic-import "^6.18.0" 1905 | babel-plugin-transform-amd-system-wrapper "^0.3.7" 1906 | babel-plugin-transform-cjs-system-wrapper "^0.6.2" 1907 | babel-plugin-transform-es2015-modules-systemjs "^6.6.5" 1908 | babel-plugin-transform-global-system-wrapper "^0.3.4" 1909 | babel-plugin-transform-system-register "^0.0.1" 1910 | bluebird "^3.3.4" 1911 | data-uri-to-buffer "0.0.4" 1912 | es6-template-strings "^2.0.0" 1913 | glob "^7.0.3" 1914 | mkdirp "^0.5.1" 1915 | rollup "^0.58.2" 1916 | source-map "^0.5.3" 1917 | systemjs "^0.19.46" 1918 | traceur "0.0.105" 1919 | uglify-js "^2.6.1" 1920 | 1921 | systemjs@^0.19.46: 1922 | version "0.19.47" 1923 | resolved "https://registry.yarnpkg.com/systemjs/-/systemjs-0.19.47.tgz#c8c93937180f3f5481c769cd2720763fb4a31c6f" 1924 | dependencies: 1925 | when "^3.7.5" 1926 | 1927 | tiny-lr@^1.1.1: 1928 | version "1.1.1" 1929 | resolved "https://registry.yarnpkg.com/tiny-lr/-/tiny-lr-1.1.1.tgz#9fa547412f238fedb068ee295af8b682c98b2aab" 1930 | dependencies: 1931 | body "^5.1.0" 1932 | debug "^3.1.0" 1933 | faye-websocket "~0.10.0" 1934 | livereload-js "^2.3.0" 1935 | object-assign "^4.1.0" 1936 | qs "^6.4.0" 1937 | 1938 | to-fast-properties@^1.0.1: 1939 | version "1.0.2" 1940 | resolved "https://registry.yarnpkg.com/to-fast-properties/-/to-fast-properties-1.0.2.tgz#f3f5c0c3ba7299a7ef99427e44633257ade43320" 1941 | 1942 | to-fast-properties@^1.0.3: 1943 | version "1.0.3" 1944 | resolved "https://registry.yarnpkg.com/to-fast-properties/-/to-fast-properties-1.0.3.tgz#b83571fa4d8c25b82e231b06e3a3055de4ca1a47" 1945 | 1946 | traceur@0.0.105: 1947 | version "0.0.105" 1948 | resolved "https://registry.yarnpkg.com/traceur/-/traceur-0.0.105.tgz#5cf9dee83d6b77861c3d6c44d53859aed7ab0479" 1949 | dependencies: 1950 | commander "2.9.x" 1951 | glob "5.0.x" 1952 | rsvp "^3.0.13" 1953 | semver "^4.3.3" 1954 | source-map-support "~0.2.8" 1955 | 1956 | trim-newlines@^1.0.0: 1957 | version "1.0.0" 1958 | resolved "https://registry.yarnpkg.com/trim-newlines/-/trim-newlines-1.0.0.tgz#5887966bb582a4503a41eb524f7d35011815a613" 1959 | 1960 | trim-right@^1.0.1: 1961 | version "1.0.1" 1962 | resolved "https://registry.yarnpkg.com/trim-right/-/trim-right-1.0.1.tgz#cb2e1203067e0c8de1f614094b9fe45704ea6003" 1963 | 1964 | typedarray@^0.0.6: 1965 | version "0.0.6" 1966 | resolved "https://registry.yarnpkg.com/typedarray/-/typedarray-0.0.6.tgz#867ac74e3864187b1d3d47d996a78ec5c8830777" 1967 | 1968 | uglify-js@^2.6.1: 1969 | version "2.6.4" 1970 | resolved "https://registry.yarnpkg.com/uglify-js/-/uglify-js-2.6.4.tgz#65ea2fb3059c9394692f15fed87c2b36c16b9adf" 1971 | dependencies: 1972 | async "~0.2.6" 1973 | source-map "~0.5.1" 1974 | uglify-to-browserify "~1.0.0" 1975 | yargs "~3.10.0" 1976 | 1977 | uglify-js@~2.8.21: 1978 | version "2.8.29" 1979 | resolved "https://registry.yarnpkg.com/uglify-js/-/uglify-js-2.8.29.tgz#29c5733148057bb4e1f75df35b7a9cb72e6a59dd" 1980 | dependencies: 1981 | source-map "~0.5.1" 1982 | yargs "~3.10.0" 1983 | optionalDependencies: 1984 | uglify-to-browserify "~1.0.0" 1985 | 1986 | uglify-to-browserify@~1.0.0: 1987 | version "1.0.2" 1988 | resolved "https://registry.yarnpkg.com/uglify-to-browserify/-/uglify-to-browserify-1.0.2.tgz#6e0924d6bda6b5afe349e39a6d632850a0f882b7" 1989 | 1990 | underscore.string@~3.3.4: 1991 | version "3.3.4" 1992 | resolved "https://registry.yarnpkg.com/underscore.string/-/underscore.string-3.3.4.tgz#2c2a3f9f83e64762fdc45e6ceac65142864213db" 1993 | dependencies: 1994 | sprintf-js "^1.0.3" 1995 | util-deprecate "^1.0.2" 1996 | 1997 | uri-path@^1.0.0: 1998 | version "1.0.0" 1999 | resolved "https://registry.yarnpkg.com/uri-path/-/uri-path-1.0.0.tgz#9747f018358933c31de0fccfd82d138e67262e32" 2000 | 2001 | util-deprecate@^1.0.2, util-deprecate@~1.0.1: 2002 | version "1.0.2" 2003 | resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf" 2004 | 2005 | validate-npm-package-license@^3.0.1: 2006 | version "3.0.3" 2007 | resolved "https://registry.yarnpkg.com/validate-npm-package-license/-/validate-npm-package-license-3.0.3.tgz#81643bcbef1bdfecd4623793dc4648948ba98338" 2008 | dependencies: 2009 | spdx-correct "^3.0.0" 2010 | spdx-expression-parse "^3.0.0" 2011 | 2012 | websocket-driver@>=0.5.1: 2013 | version "0.7.0" 2014 | resolved "https://registry.yarnpkg.com/websocket-driver/-/websocket-driver-0.7.0.tgz#0caf9d2d755d93aee049d4bdd0d3fe2cca2a24eb" 2015 | dependencies: 2016 | http-parser-js ">=0.4.0" 2017 | websocket-extensions ">=0.1.1" 2018 | 2019 | websocket-extensions@>=0.1.1: 2020 | version "0.1.3" 2021 | resolved "https://registry.yarnpkg.com/websocket-extensions/-/websocket-extensions-0.1.3.tgz#5d2ff22977003ec687a4b87073dfbbac146ccf29" 2022 | 2023 | when@^3.7.5: 2024 | version "3.7.8" 2025 | resolved "https://registry.yarnpkg.com/when/-/when-3.7.8.tgz#c7130b6a7ea04693e842cdc9e7a1f2aa39a39f82" 2026 | 2027 | which@~1.3.0: 2028 | version "1.3.1" 2029 | resolved "https://registry.yarnpkg.com/which/-/which-1.3.1.tgz#a45043d54f5805316da8d62f9f50918d3da70b0a" 2030 | dependencies: 2031 | isexe "^2.0.0" 2032 | 2033 | window-size@0.1.0: 2034 | version "0.1.0" 2035 | resolved "https://registry.yarnpkg.com/window-size/-/window-size-0.1.0.tgz#5438cd2ea93b202efa3a19fe8887aee7c94f9c9d" 2036 | 2037 | wordwrap@0.0.2: 2038 | version "0.0.2" 2039 | resolved "https://registry.yarnpkg.com/wordwrap/-/wordwrap-0.0.2.tgz#b79669bb42ecb409f83d583cad52ca17eaa1643f" 2040 | 2041 | wrappy@1: 2042 | version "1.0.2" 2043 | resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f" 2044 | 2045 | xtend@~4.0.0: 2046 | version "4.0.1" 2047 | resolved "https://registry.yarnpkg.com/xtend/-/xtend-4.0.1.tgz#a5c6d532be656e23db820efb943a1f04998d63af" 2048 | 2049 | yargs@~3.10.0: 2050 | version "3.10.0" 2051 | resolved "https://registry.yarnpkg.com/yargs/-/yargs-3.10.0.tgz#f7ee7bd857dd7c1d2d38c0e74efbd681d1431fd1" 2052 | dependencies: 2053 | camelcase "^1.0.2" 2054 | cliui "^2.1.0" 2055 | decamelize "^1.0.0" 2056 | window-size "0.1.0" 2057 | --------------------------------------------------------------------------------