UNPKG

26.9 kBJavaScriptView Raw
1'use strict';
2
3Object.defineProperty(exports, "__esModule", {
4 value: true
5});
6
7var _promise = require('babel-runtime/core-js/promise');
8
9var _promise2 = _interopRequireDefault(_promise);
10
11var _regenerator = require('babel-runtime/regenerator');
12
13var _regenerator2 = _interopRequireDefault(_regenerator);
14
15var _asyncToGenerator2 = require('babel-runtime/helpers/asyncToGenerator');
16
17var _asyncToGenerator3 = _interopRequireDefault(_asyncToGenerator2);
18
19var _getPrototypeOf = require('babel-runtime/core-js/object/get-prototype-of');
20
21var _getPrototypeOf2 = _interopRequireDefault(_getPrototypeOf);
22
23var _classCallCheck2 = require('babel-runtime/helpers/classCallCheck');
24
25var _classCallCheck3 = _interopRequireDefault(_classCallCheck2);
26
27var _createClass2 = require('babel-runtime/helpers/createClass');
28
29var _createClass3 = _interopRequireDefault(_createClass2);
30
31var _possibleConstructorReturn2 = require('babel-runtime/helpers/possibleConstructorReturn');
32
33var _possibleConstructorReturn3 = _interopRequireDefault(_possibleConstructorReturn2);
34
35var _inherits2 = require('babel-runtime/helpers/inherits');
36
37var _inherits3 = _interopRequireDefault(_inherits2);
38
39var _cliEngineCommand = require('cli-engine-command');
40
41var _path = require('path');
42
43var _path2 = _interopRequireDefault(_path);
44
45var _rwlockfile = require('rwlockfile');
46
47var _rwlockfile2 = _interopRequireDefault(_rwlockfile);
48
49function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
50
51var _class = function (_Base) {
52 (0, _inherits3.default)(_class, _Base);
53
54 function _class() {
55 (0, _classCallCheck3.default)(this, _class);
56 return (0, _possibleConstructorReturn3.default)(this, (_class.__proto__ || (0, _getPrototypeOf2.default)(_class)).apply(this, arguments));
57 }
58
59 (0, _createClass3.default)(_class, [{
60 key: 'fetchManifest',
61 value: function () {
62 var _ref = (0, _asyncToGenerator3.default)(_regenerator2.default.mark(function _callee(channel) {
63 var url, manifest;
64 return _regenerator2.default.wrap(function _callee$(_context) {
65 while (1) {
66 switch (_context.prev = _context.next) {
67 case 0:
68 if (this.config.s3.host) {
69 _context.next = 2;
70 break;
71 }
72
73 throw new Error('S3 host not defined');
74
75 case 2:
76 _context.prev = 2;
77 url = 'https://' + this.config.s3.host + '/' + this.config.name + '/channels/' + channel + '/' + process.platform + '-' + process.arch;
78 _context.next = 6;
79 return this.http.get(url);
80
81 case 6:
82 manifest = _context.sent;
83 return _context.abrupt('return', manifest);
84
85 case 10:
86 _context.prev = 10;
87 _context.t0 = _context['catch'](2);
88
89 if (!(_context.t0.statusCode === 403)) {
90 _context.next = 14;
91 break;
92 }
93
94 throw new Error('HTTP 403: Invalid channel ' + channel);
95
96 case 14:
97 throw _context.t0;
98
99 case 15:
100 case 'end':
101 return _context.stop();
102 }
103 }
104 }, _callee, this, [[2, 10]]);
105 }));
106
107 function fetchManifest(_x) {
108 return _ref.apply(this, arguments);
109 }
110
111 return fetchManifest;
112 }()
113 }, {
114 key: 'update',
115 value: function () {
116 var _ref2 = (0, _asyncToGenerator3.default)(_regenerator2.default.mark(function _callee2(manifest) {
117 var url, stream, dir, tmp;
118 return _regenerator2.default.wrap(function _callee2$(_context2) {
119 while (1) {
120 switch (_context2.prev = _context2.next) {
121 case 0:
122 if (this.config.s3.host) {
123 _context2.next = 2;
124 break;
125 }
126
127 throw new Error('S3 host not defined');
128
129 case 2:
130 url = 'https://' + this.config.s3.host + '/' + this.config.name + '/channels/' + manifest.channel + '/' + this.base(manifest) + '.tar.gz';
131 _context2.next = 5;
132 return this.http.stream(url);
133
134 case 5:
135 stream = _context2.sent;
136 dir = _path2.default.join(this.config.dirs.data, 'cli');
137 tmp = _path2.default.join(this.config.dirs.data, 'cli_tmp');
138 _context2.next = 10;
139 return this.extract(stream, tmp);
140
141 case 10:
142 _context2.next = 12;
143 return _rwlockfile2.default.write(this.updatelockfile, { skipOwnPid: true });
144
145 case 12:
146 this.fs.removeSync(dir);
147 this.fs.renameSync(_path2.default.join(tmp, this.base(manifest)), dir);
148 this.fs.removeSync(tmp);
149
150 case 15:
151 case 'end':
152 return _context2.stop();
153 }
154 }
155 }, _callee2, this);
156 }));
157
158 function update(_x2) {
159 return _ref2.apply(this, arguments);
160 }
161
162 return update;
163 }()
164 }, {
165 key: 'extract',
166 value: function extract(stream, dir) {
167 var _this2 = this;
168
169 var zlib = require('zlib');
170 var tar = require('tar-stream');
171
172 return new _promise2.default(function (resolve) {
173 _this2.fs.removeSync(dir);
174 var extract = tar.extract();
175 extract.on('entry', function (header, stream, next) {
176 var p = _path2.default.join(dir, header.name);
177 var opts = { mode: header.mode };
178 switch (header.type) {
179 case 'directory':
180 _this2.fs.mkdirpSync(p, opts);
181 next();
182 break;
183 case 'file':
184 stream.pipe(_this2.fs.createWriteStream(p, opts));
185 break;
186 case 'symlink':
187 // ignore symlinks since they will not work on windows
188 next();
189 break;
190 default:
191 throw new Error(header.type);
192 }
193 stream.resume();
194 stream.on('end', next);
195 });
196 extract.on('finish', resolve);
197 stream.pipe(zlib.createGunzip()).pipe(extract);
198 });
199 }
200 }, {
201 key: 'base',
202 value: function base(manifest) {
203 return this.config.name + '-v' + manifest.version + '-' + process.platform + '-' + process.arch;
204 }
205 }, {
206 key: 'restartCLI',
207 value: function () {
208 var _ref3 = (0, _asyncToGenerator3.default)(_regenerator2.default.mark(function _callee3() {
209 var _require, spawnSync, _spawnSync, status;
210
211 return _regenerator2.default.wrap(function _callee3$(_context3) {
212 while (1) {
213 switch (_context3.prev = _context3.next) {
214 case 0:
215 _context3.next = 2;
216 return _rwlockfile2.default.read(this.updatelockfile);
217
218 case 2:
219 _rwlockfile2.default.unreadSync(this.updatelockfile);
220 _require = require('child_process'), spawnSync = _require.spawnSync;
221
222 if (this.binPath) {
223 _context3.next = 6;
224 break;
225 }
226
227 return _context3.abrupt('return');
228
229 case 6:
230 _spawnSync = spawnSync(this.binPath, process.argv.slice(2), { stdio: 'inherit', shell: true }), status = _spawnSync.status;
231
232 this.exit(status);
233
234 case 8:
235 case 'end':
236 return _context3.stop();
237 }
238 }
239 }, _callee3, this);
240 }));
241
242 function restartCLI() {
243 return _ref3.apply(this, arguments);
244 }
245
246 return restartCLI;
247 }()
248 }, {
249 key: 'autoupdate',
250 value: function () {
251 var _ref4 = (0, _asyncToGenerator3.default)(_regenerator2.default.mark(function _callee4() {
252 var _this3 = this;
253
254 var fd, _require2, spawn;
255
256 return _regenerator2.default.wrap(function _callee4$(_context4) {
257 while (1) {
258 switch (_context4.prev = _context4.next) {
259 case 0:
260 _context4.prev = 0;
261
262 if (this.autoupdateNeeded) {
263 _context4.next = 3;
264 break;
265 }
266
267 return _context4.abrupt('return');
268
269 case 3:
270 this.fs.writeFileSync(this.autoupdatefile, '');
271
272 if (!this.config.updateDisabled) {
273 _context4.next = 7;
274 break;
275 }
276
277 _context4.next = 7;
278 return this.warnIfUpdateAvailable();
279
280 case 7:
281 _context4.next = 9;
282 return this.checkIfUpdating();
283
284 case 9:
285 fd = this.fs.openSync(this.autoupdatelogfile, 'a');
286
287 if (this.binPath) {
288 _context4.next = 12;
289 break;
290 }
291
292 return _context4.abrupt('return');
293
294 case 12:
295 _require2 = require('child_process'), spawn = _require2.spawn;
296
297 spawn(this.binPath, ['update'], { stdio: [null, fd, fd], detached: true }).on('error', function (e) {
298 return _this3.warn(e, 'autoupdate:');
299 });
300 _context4.next = 19;
301 break;
302
303 case 16:
304 _context4.prev = 16;
305 _context4.t0 = _context4['catch'](0);
306 this.warn(_context4.t0, 'autoupdate:');
307 case 19:
308 case 'end':
309 return _context4.stop();
310 }
311 }
312 }, _callee4, this, [[0, 16]]);
313 }));
314
315 function autoupdate() {
316 return _ref4.apply(this, arguments);
317 }
318
319 return autoupdate;
320 }()
321 }, {
322 key: 'warnIfUpdateAvailable',
323 value: function () {
324 var _ref5 = (0, _asyncToGenerator3.default)(_regenerator2.default.mark(function _callee5() {
325 var manifest, local, remote;
326 return _regenerator2.default.wrap(function _callee5$(_context5) {
327 while (1) {
328 switch (_context5.prev = _context5.next) {
329 case 0:
330 _context5.next = 2;
331 return this.fetchManifest(this.config.channel);
332
333 case 2:
334 manifest = _context5.sent;
335 local = this.config.version.split('.');
336 remote = manifest.version.split('.');
337
338 if (local[0] !== remote[0] || local[1] !== remote[1]) {
339 this.stderr.log(this.config.name + ': update available from ' + this.config.version + ' to ' + manifest.version);
340 }
341
342 case 6:
343 case 'end':
344 return _context5.stop();
345 }
346 }
347 }, _callee5, this);
348 }));
349
350 function warnIfUpdateAvailable() {
351 return _ref5.apply(this, arguments);
352 }
353
354 return warnIfUpdateAvailable;
355 }()
356 }, {
357 key: 'checkIfUpdating',
358 value: function () {
359 var _ref6 = (0, _asyncToGenerator3.default)(_regenerator2.default.mark(function _callee6() {
360 var lock;
361 return _regenerator2.default.wrap(function _callee6$(_context6) {
362 while (1) {
363 switch (_context6.prev = _context6.next) {
364 case 0:
365 lock = require('rwlockfile');
366 _context6.next = 3;
367 return lock.hasWriter(this.updatelockfile);
368
369 case 3:
370 if (!_context6.sent) {
371 _context6.next = 9;
372 break;
373 }
374
375 this.warn(this.config.name + ': warning: update in process');
376 _context6.next = 7;
377 return this.restartCLI();
378
379 case 7:
380 _context6.next = 11;
381 break;
382
383 case 9:
384 _context6.next = 11;
385 return lock.read(this.updatelockfile);
386
387 case 11:
388 case 'end':
389 return _context6.stop();
390 }
391 }
392 }, _callee6, this);
393 }));
394
395 function checkIfUpdating() {
396 return _ref6.apply(this, arguments);
397 }
398
399 return checkIfUpdating;
400 }()
401 }, {
402 key: 'autoupdatefile',
403 get: function get() {
404 return _path2.default.join(this.config.dirs.cache, 'autoupdate');
405 }
406 }, {
407 key: 'autoupdatelogfile',
408 get: function get() {
409 return _path2.default.join(this.config.dirs.cache, 'autoupdate.log');
410 }
411 }, {
412 key: 'updatelockfile',
413 get: function get() {
414 return _path2.default.join(this.config.dirs.cache, 'update.lock');
415 }
416 }, {
417 key: 'binPath',
418 get: function get() {
419 return process.env.CLI_BINPATH;
420 }
421 }, {
422 key: 'autoupdateNeeded',
423 get: function get() {
424 try {
425 var moment = require('moment');
426 var stat = this.fs.statSync(this.autoupdatefile);
427 return moment(stat.mtime).isBefore(moment().subtract(4, 'hours'));
428 } catch (err) {
429 if (err.code !== 'ENOENT') console.error(err.stack);
430 return true;
431 }
432 }
433 }]);
434 return _class;
435}(_cliEngineCommand.Base);
436/* globals
437 stream$Readable
438*/
439
440exports.default = _class;
441//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/updater.js"],"names":["channel","config","s3","host","Error","url","name","process","platform","arch","http","get","manifest","statusCode","base","stream","dir","join","dirs","data","tmp","extract","write","updatelockfile","skipOwnPid","fs","removeSync","renameSync","zlib","require","tar","on","header","next","p","opts","mode","type","mkdirpSync","pipe","createWriteStream","resume","resolve","createGunzip","version","read","unreadSync","spawnSync","binPath","argv","slice","stdio","shell","status","exit","autoupdateNeeded","writeFileSync","autoupdatefile","updateDisabled","warnIfUpdateAvailable","checkIfUpdating","fd","openSync","autoupdatelogfile","spawn","detached","warn","e","fetchManifest","local","split","remote","stderr","log","lock","hasWriter","restartCLI","cache","env","CLI_BINPATH","moment","stat","statSync","mtime","isBefore","subtract","err","code","console","error","stack"],"mappings":"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","file":"updater.js","sourcesContent":["// @flow\n/* globals\n   stream$Readable\n*/\n\nimport {Base} from 'cli-engine-command'\nimport path from 'path'\nimport lock from 'rwlockfile'\n\ntype Manifest = {\n  version: string,\n  channel: string,\n  sha256: string\n}\n\nexport default class extends Base {\n  get autoupdatefile (): string { return path.join(this.config.dirs.cache, 'autoupdate') }\n  get autoupdatelogfile (): string { return path.join(this.config.dirs.cache, 'autoupdate.log') }\n  get updatelockfile (): string { return path.join(this.config.dirs.cache, 'update.lock') }\n  get binPath (): ?string { return process.env.CLI_BINPATH }\n\n  async fetchManifest (channel: string): Promise<Manifest> {\n    if (!this.config.s3.host) throw new Error('S3 host not defined')\n    try {\n      let url = `https://${this.config.s3.host}/${this.config.name}/channels/${channel}/${process.platform}-${process.arch}`\n      let manifest = await this.http.get(url)\n      return ((manifest: any): Promise<Manifest>)\n    } catch (err) {\n      if (err.statusCode === 403) throw new Error(`HTTP 403: Invalid channel ${channel}`)\n      throw err\n    }\n  }\n\n  async update (manifest: Manifest) {\n    if (!this.config.s3.host) throw new Error('S3 host not defined')\n    let url = `https://${this.config.s3.host}/${this.config.name}/channels/${manifest.channel}/${this.base(manifest)}.tar.gz`\n    let stream = await this.http.stream(url)\n    let dir = path.join(this.config.dirs.data, 'cli')\n    let tmp = path.join(this.config.dirs.data, 'cli_tmp')\n    await this.extract(stream, tmp)\n    await lock.write(this.updatelockfile, {skipOwnPid: true})\n    this.fs.removeSync(dir)\n    this.fs.renameSync(path.join(tmp, this.base(manifest)), dir)\n    this.fs.removeSync(tmp)\n  }\n\n  extract (stream: stream$Readable, dir: string) {\n    const zlib = require('zlib')\n    const tar = require('tar-stream')\n\n    return new Promise(resolve => {\n      this.fs.removeSync(dir)\n      let extract = tar.extract()\n      extract.on('entry', (header, stream, next) => {\n        let p = path.join(dir, header.name)\n        let opts = {mode: header.mode}\n        switch (header.type) {\n          case 'directory':\n            this.fs.mkdirpSync(p, opts)\n            next()\n            break\n          case 'file':\n            stream.pipe(this.fs.createWriteStream(p, opts))\n            break\n          case 'symlink':\n            // ignore symlinks since they will not work on windows\n            next()\n            break\n          default: throw new Error(header.type)\n        }\n        stream.resume()\n        stream.on('end', next)\n      })\n      extract.on('finish', resolve)\n      stream\n      .pipe(zlib.createGunzip())\n      .pipe(extract)\n    })\n  }\n\n  base (manifest: Manifest): string {\n    return `${this.config.name}-v${manifest.version}-${process.platform}-${process.arch}`\n  }\n\n  async restartCLI () {\n    await lock.read(this.updatelockfile)\n    lock.unreadSync(this.updatelockfile)\n    const {spawnSync} = require('child_process')\n    if (!this.binPath) return\n    const {status} = spawnSync(this.binPath, process.argv.slice(2), {stdio: 'inherit', shell: true})\n    this.exit(status)\n  }\n\n  get autoupdateNeeded () {\n    try {\n      const moment = require('moment')\n      const stat = this.fs.statSync(this.autoupdatefile)\n      return moment(stat.mtime).isBefore(moment().subtract(4, 'hours'))\n    } catch (err) {\n      if (err.code !== 'ENOENT') console.error(err.stack)\n      return true\n    }\n  }\n\n  async autoupdate () {\n    try {\n      if (!this.autoupdateNeeded) return\n      this.fs.writeFileSync(this.autoupdatefile, '')\n      if (this.config.updateDisabled) await this.warnIfUpdateAvailable()\n      await this.checkIfUpdating()\n      let fd = this.fs.openSync(this.autoupdatelogfile, 'a')\n      if (!this.binPath) return\n      const {spawn} = require('child_process')\n      spawn(this.binPath, ['update'], {stdio: [null, fd, fd], detached: true})\n      .on('error', e => this.warn(e, 'autoupdate:'))\n    } catch (e) { this.warn(e, 'autoupdate:') }\n  }\n\n  async warnIfUpdateAvailable () {\n    const manifest = await this.fetchManifest(this.config.channel)\n    let local = this.config.version.split('.')\n    let remote = manifest.version.split('.')\n    if (local[0] !== remote[0] || local[1] !== remote[1]) {\n      this.stderr.log(`${this.config.name}: update available from ${this.config.version} to ${manifest.version}`)\n    }\n  }\n\n  async checkIfUpdating () {\n    const lock = require('rwlockfile')\n    if (await lock.hasWriter(this.updatelockfile)) {\n      this.warn(`${this.config.name}: warning: update in process`)\n      await this.restartCLI()\n    } else await lock.read(this.updatelockfile)\n  }\n}\n"]}
\No newline at end of file