UNPKG

booletwa

Version:

Generate TWA projects from a Web Manifest

98 lines (81 loc) 3.3 kB
/* * Copyright 2021 Google Inc. All Rights Reserved. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import * as fs from 'fs'; import {context, Response as FetchH2Response} from 'fetch-h2'; import nodefetch, {Response as NodeFetchResponse} from 'node-fetch'; export type FetchEngine = 'node-fetch' | 'fetch-h2'; const DEFAULT_FETCH_ENGINE: FetchEngine = 'fetch-h2'; export type NodeFetchOrFetchH2Response = FetchH2Response | NodeFetchResponse; const userAgent = 'Mozilla/5.0 (X11; Linux x86_64; rv:60.0) Gecko/20100101 Firefox/81.0'; const fetchh2Ctx = context({userAgent: userAgent, overwriteUserAgent: true}); const fetchh2 = fetchh2Ctx.fetch; class FetchUtils { private fetchEngine = DEFAULT_FETCH_ENGINE; setFetchEngine(newFetchEngine: FetchEngine): void { this.fetchEngine = newFetchEngine; } async fetch(input: string): Promise<NodeFetchOrFetchH2Response> { if (this.fetchEngine == 'node-fetch') { return await nodefetch(input, {redirect: 'follow', headers: {'User-Agent': userAgent}}); } else { return await fetchh2(input, {redirect: 'follow'}); } } /** * Downloads a file from `url` and saves it to `path`. If a `progressCallback` function is passed, it * will be invoked for every chunk received. If the value of `total` parameter is -1, it means we * were unable to determine to total file size before starting the download. */ async downloadFile( url: string, path: string, progressCallback?: (current: number, total: number) => void, ): Promise<void> { let result; let readableStream: NodeJS.ReadableStream; if (this.fetchEngine === 'node-fetch') { result = await nodefetch(url); readableStream = result.body; } else { result = await fetchh2(url, {redirect: 'follow'}); readableStream = await result.readable(); } // Try to determine the file size via the `Content-Length` header. This may not be available // for all cases. const contentLength = result.headers.get('content-length'); const fileSize = contentLength ? parseInt(contentLength) : -1; const fileStream = fs.createWriteStream(path); let received = 0; await new Promise((resolve, reject) => { readableStream.pipe(fileStream); // Even though we're piping the chunks, we intercept them to check for the download progress. if (progressCallback) { readableStream.on('data', (chunk) => { received = received + chunk.length; progressCallback(received, fileSize); }); } readableStream.on('error', (err) => { reject(err); }); fileStream.on('finish', () => { resolve(); }); }); } } const fetchUtils = new FetchUtils(); export {fetchUtils};