mizzkey/packages/backend/src/core/DownloadService.ts

98 lines
2.6 KiB
TypeScript
Raw Normal View History

2022-09-18 03:27:08 +09:00
import * as fs from 'node:fs';
import * as stream from 'node:stream';
import * as util from 'node:util';
import { Inject, Injectable } from '@nestjs/common';
import IPCIDR from 'ip-cidr';
import PrivateIp from 'private-ip';
import got, * as Got from 'got';
import chalk from 'chalk';
import { DI } from '@/di-symbols.js';
2022-09-21 05:33:11 +09:00
import type { Config } from '@/config.js';
2022-09-18 03:27:08 +09:00
import { HttpRequestService } from '@/core/HttpRequestService.js';
import { createTemp } from '@/misc/create-temp.js';
import { StatusError } from '@/misc/status-error.js';
2022-09-18 23:07:41 +09:00
import { LoggerService } from '@/core/LoggerService.js';
import type Logger from '@/logger.js';
2022-09-18 03:27:08 +09:00
const pipeline = util.promisify(stream.pipeline);
import { bindThis } from '@/decorators.js';
2022-09-18 03:27:08 +09:00
@Injectable()
export class DownloadService {
2022-09-19 03:11:50 +09:00
private logger: Logger;
2022-09-18 03:27:08 +09:00
constructor(
@Inject(DI.config)
private config: Config,
private httpRequestService: HttpRequestService,
2022-09-18 23:07:41 +09:00
private loggerService: LoggerService,
2022-09-18 03:27:08 +09:00
) {
2022-09-19 03:11:50 +09:00
this.logger = this.loggerService.getLogger('download');
2022-09-18 03:27:08 +09:00
}
@bindThis
2022-09-18 03:27:08 +09:00
public async downloadUrl(url: string, path: string): Promise<void> {
this.logger.info(`Downloading ${chalk.cyan(url)} to ${chalk.cyanBright(path)} ...`);
2023-01-02 18:42:00 +00:00
2022-09-18 03:27:08 +09:00
const timeout = 30 * 1000;
const operationTimeout = 60 * 1000;
const maxSize = this.config.maxFileSize ?? 262144000;
2023-01-02 18:42:00 +00:00
const response = await this.httpRequestService.fetch({
method: 'GET',
url,
2022-09-18 03:27:08 +09:00
headers: {
'User-Agent': this.config.userAgent,
},
2023-01-02 18:42:00 +00:00
timeout,
size: maxSize,
ipCheckers:
(process.env.NODE_ENV === 'production' || process.env.NODE_ENV === 'test') &&
!this.config.proxy ?
[{ type: 'black', fn: this.isPrivateIp }] :
undefined,
// http2: false, // default
2022-09-18 03:27:08 +09:00
});
2023-01-02 18:47:25 +00:00
if (response.body === null) {
throw new StatusError('No body', 400, 'No body');
2022-09-18 03:27:08 +09:00
}
2023-01-02 18:47:25 +00:00
await pipeline(stream.Readable.fromWeb(response.body), fs.createWriteStream(path));
2022-09-19 03:11:50 +09:00
this.logger.succ(`Download finished: ${chalk.cyan(url)}`);
2022-09-18 03:27:08 +09:00
}
@bindThis
2022-09-18 03:27:08 +09:00
public async downloadTextFile(url: string): Promise<string> {
// Create temp file
const [path, cleanup] = await createTemp();
2022-09-19 03:11:50 +09:00
this.logger.info(`text file: Temp file is ${path}`);
2022-09-18 03:27:08 +09:00
try {
// write content at URL to temp file
await this.downloadUrl(url, path);
const text = await util.promisify(fs.readFile)(path, 'utf8');
return text;
} finally {
cleanup();
}
}
@bindThis
2022-09-19 03:11:50 +09:00
private isPrivateIp(ip: string): boolean {
2022-09-18 03:27:08 +09:00
for (const net of this.config.allowedPrivateNetworks ?? []) {
const cidr = new IPCIDR(net);
if (cidr.contains(ip)) {
return false;
}
}
2023-01-02 18:42:00 +00:00
return PrivateIp(ip) ?? false;
2022-09-18 03:27:08 +09:00
}
}