| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200 |
- import { dirname, basename, resolve } from 'path';
- import * as request from 'request';
- import * as ProgressBar from 'progress';
- import { ensureDirSync, exists, lstatAsync, writeFile } from 'fs-extra-promise';
- const debug = require('debug')('build:downloader');
- const progress = require('request-progress');
- import { Event } from './Event';
- import { mergeOptions, extractGeneric } from '../util';
- const DIR_CACHES = resolve(dirname(module.filename), '..', '..', '..', 'caches');
- ensureDirSync(DIR_CACHES);
- interface IRequestProgress {
- percent: number;
- speed: number;
- size: {
- total: number,
- transferred: number,
- };
- time: {
- elapsed: number,
- remaining: number,
- };
- }
- export abstract class DownloaderBase {
- public onProgress: Event<IRequestProgress> = new Event('progress');
- protected destination: string = DIR_CACHES;
- public abstract async fetch(): Promise<string>;
- protected abstract handleVersion(version: string): Promise<string>;
- public async fetchAndExtract() {
- const archive = await this.fetch();
- const dest = `${ archive }-extracted`;
- await extractGeneric(archive, dest);
- return dest;
- }
- protected getVersions(): Promise<any> {
- return new Promise((resolve, reject) => {
- request('https://nwjs.io/versions.json', (err, res, body) => {
- if(err) {
- return reject(err);
- }
- const json = JSON.parse(body);
- resolve(json);
- });
- });
- }
- protected setDestination(destination: string) {
- this.destination = destination;
- }
- protected handlePlatform(platform: string) {
- switch(platform) {
- case 'win32':
- case 'win':
- return 'win';
- case 'darwin':
- case 'osx':
- case 'mac':
- return 'osx';
- case 'linux':
- return 'linux';
- default:
- throw new Error('ERROR_UNKNOWN_PLATFORM');
- }
- }
- protected handleArch(arch: string) {
- switch(arch) {
- case 'x86':
- case 'ia32':
- return 'ia32';
- case 'x64':
- return 'x64';
- default:
- throw new Error('ERROR_UNKNOWN_PLATFORM');
- }
- }
- protected getLocalSize(path: string): Promise<number> {
- return lstatAsync(path)
- .then(stat => stat.size);
- }
- protected getRemoteSize(url: string): Promise<number> {
- return new Promise((resolve, reject) => {
- request.head(url)
- .on('error', reject)
- .on('response', res => resolve(parseInt(res.headers['content-length'], 10)));
- });
- }
- protected isFileExists(path: string) {
- return new Promise((resolve, reject) => {
- exists(path, resolve);
- });
- }
- protected async isFileSynced(url: string, path: string) {
- const localSize = await this.getLocalSize(path);
- const remoteSize = await this.getRemoteSize(url);
- debug('in isFileSynced', 'localSize', localSize);
- debug('in isFileSynced', 'remoteSize', remoteSize);
- return localSize == remoteSize;
- }
- protected async download(url: string, filename: string, path: string, showProgress: boolean) {
- let bar: ProgressBar = null;
- const onProgress = (state: IRequestProgress) => {
- if(!state.size.total) {
- return;
- }
- if(!bar) {
- bar = new ProgressBar('[:bar] :speedKB/s :etas', {
- width: 50,
- total: state.size.total,
- });
- console.info('');
- }
- bar.update(state.size.transferred / state.size.total, {
- speed: (state.speed / 1000).toFixed(2),
- });
- };
- if(showProgress) {
- this.onProgress.subscribe(onProgress);
- }
- debug('in download', 'start downloading', filename);
- await new Promise((resolve, reject) => {
- progress(request(url, {
- encoding: null,
- }, (err, res, data) => {
- if(err) {
- return reject(err);
- }
- if(res.statusCode != 200) {
- const e = new Error(`ERROR_STATUS_CODE statusCode = ${ res.statusCode }`);
- return reject(e);
- }
- writeFile(path, data, err => err ? reject(err) : resolve());
- }))
- .on('progress', (state: IRequestProgress) => {
- this.onProgress.trigger(state);
- });
- });
- debug('in fetch', 'end downloading', filename);
- if(showProgress) {
- this.onProgress.unsubscribe(onProgress);
- if(bar) {
- console.info('');
- bar.terminate();
- }
- }
- return path;
- }
- }
|