• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1"use strict";
2var __importDefault = (this && this.__importDefault) || function (mod) {
3    return (mod && mod.__esModule) ? mod : { "default": mod };
4};
5Object.defineProperty(exports, "__esModule", { value: true });
6exports.DefaultFetcher = exports.BaseFetcher = void 0;
7const debug_1 = __importDefault(require("debug"));
8const fs_1 = __importDefault(require("fs"));
9const make_fetch_happen_1 = __importDefault(require("make-fetch-happen"));
10const util_1 = __importDefault(require("util"));
11const error_1 = require("./error");
12const tmpfile_1 = require("./utils/tmpfile");
13const log = (0, debug_1.default)('tuf:fetch');
14class BaseFetcher {
15    // Download file from given URL. The file is downloaded to a temporary
16    // location and then passed to the given handler. The handler is responsible
17    // for moving the file to its final location. The temporary file is deleted
18    // after the handler returns.
19    async downloadFile(url, maxLength, handler) {
20        return (0, tmpfile_1.withTempFile)(async (tmpFile) => {
21            const reader = await this.fetch(url);
22            let numberOfBytesReceived = 0;
23            const fileStream = fs_1.default.createWriteStream(tmpFile);
24            // Read the stream a chunk at a time so that we can check
25            // the length of the file as we go
26            try {
27                for await (const chunk of reader) {
28                    const bufferChunk = Buffer.from(chunk);
29                    numberOfBytesReceived += bufferChunk.length;
30                    if (numberOfBytesReceived > maxLength) {
31                        throw new error_1.DownloadLengthMismatchError('Max length reached');
32                    }
33                    await writeBufferToStream(fileStream, bufferChunk);
34                }
35            }
36            finally {
37                // Make sure we always close the stream
38                await util_1.default.promisify(fileStream.close).bind(fileStream)();
39            }
40            return handler(tmpFile);
41        });
42    }
43    // Download bytes from given URL.
44    async downloadBytes(url, maxLength) {
45        return this.downloadFile(url, maxLength, async (file) => {
46            const stream = fs_1.default.createReadStream(file);
47            const chunks = [];
48            for await (const chunk of stream) {
49                chunks.push(chunk);
50            }
51            return Buffer.concat(chunks);
52        });
53    }
54}
55exports.BaseFetcher = BaseFetcher;
56class DefaultFetcher extends BaseFetcher {
57    constructor(options = {}) {
58        super();
59        this.timeout = options.timeout;
60        this.retries = options.retries;
61    }
62    async fetch(url) {
63        log('GET %s', url);
64        const response = await (0, make_fetch_happen_1.default)(url, {
65            timeout: this.timeout,
66            retry: this.retries,
67        });
68        if (!response.ok || !response?.body) {
69            throw new error_1.DownloadHTTPError('Failed to download', response.status);
70        }
71        return response.body;
72    }
73}
74exports.DefaultFetcher = DefaultFetcher;
75const writeBufferToStream = async (stream, buffer) => {
76    return new Promise((resolve, reject) => {
77        stream.write(buffer, (err) => {
78            if (err) {
79                reject(err);
80            }
81            resolve(true);
82        });
83    });
84};
85