2023-01-13 05:40:33 +01:00
|
|
|
import * as fs from "node:fs";
|
2024-04-23 22:19:03 +02:00
|
|
|
import * as stream from "node:stream/promises";
|
2023-12-05 08:12:10 +01:00
|
|
|
import got, * as Got from "got";
|
2024-04-19 19:07:08 +02:00
|
|
|
import { config } from "@/config.js";
|
2024-04-13 01:41:11 +02:00
|
|
|
import { getAgentByHostname, StatusError } from "./fetch.js";
|
2023-11-26 21:33:46 +01:00
|
|
|
import chalk from "chalk";
|
2023-12-05 08:12:10 +01:00
|
|
|
import Logger from "@/services/logger.js";
|
2023-01-13 05:40:33 +01:00
|
|
|
import IPCIDR from "ip-cidr";
|
|
|
|
import PrivateIp from "private-ip";
|
2024-03-26 13:05:29 +01:00
|
|
|
import { isValidUrl } from "./is-valid-url.js";
|
2019-03-20 20:50:44 +01:00
|
|
|
|
2021-12-10 10:24:26 +01:00
|
|
|
export async function downloadUrl(url: string, path: string): Promise<void> {
|
2024-03-26 13:05:29 +01:00
|
|
|
if (!isValidUrl(url)) {
|
|
|
|
throw new StatusError("Invalid URL", 400);
|
|
|
|
}
|
|
|
|
|
2024-04-25 04:08:45 +02:00
|
|
|
const downloadLogger = new Logger("download");
|
2019-03-20 20:50:44 +01:00
|
|
|
|
2024-04-25 04:08:45 +02:00
|
|
|
downloadLogger.debug(`Downloading ${chalk.cyan(url)} ...`);
|
2020-04-09 16:42:23 +02:00
|
|
|
|
2021-09-03 14:00:44 +02:00
|
|
|
const timeout = 30 * 1000;
|
|
|
|
const operationTimeout = 60 * 1000;
|
2021-09-04 13:33:14 +02:00
|
|
|
const maxSize = config.maxFileSize || 262144000;
|
2021-09-03 14:00:44 +02:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
const req = got
|
2023-07-17 00:32:32 +02:00
|
|
|
.stream(url, {
|
2023-01-13 05:40:33 +01:00
|
|
|
headers: {
|
|
|
|
"User-Agent": config.userAgent,
|
2023-07-17 00:32:32 +02:00
|
|
|
Host: new URL(url).hostname,
|
2023-01-13 05:40:33 +01:00
|
|
|
},
|
|
|
|
timeout: {
|
|
|
|
lookup: timeout,
|
|
|
|
connect: timeout,
|
|
|
|
secureConnect: timeout,
|
|
|
|
socket: timeout, // read timeout
|
|
|
|
response: timeout,
|
|
|
|
send: timeout,
|
|
|
|
request: operationTimeout, // whole operation timeout
|
|
|
|
},
|
2024-04-13 01:41:11 +02:00
|
|
|
agent: getAgentByHostname(new URL(url).hostname),
|
2023-01-13 05:40:33 +01:00
|
|
|
http2: false, // default
|
|
|
|
retry: {
|
|
|
|
limit: 0,
|
|
|
|
},
|
|
|
|
})
|
2024-03-26 13:05:29 +01:00
|
|
|
.on("redirect", (res: Got.Response, opts: Got.NormalizedOptions) => {
|
|
|
|
if (!isValidUrl(opts.url)) {
|
2024-04-25 04:08:45 +02:00
|
|
|
downloadLogger.warn(`Invalid URL: ${opts.url}`);
|
2024-03-26 13:05:29 +01:00
|
|
|
req.destroy();
|
|
|
|
}
|
|
|
|
})
|
2023-01-13 05:40:33 +01:00
|
|
|
.on("response", (res: Got.Response) => {
|
|
|
|
if (
|
|
|
|
(process.env.NODE_ENV === "production" ||
|
|
|
|
process.env.NODE_ENV === "test") &&
|
|
|
|
!config.proxy &&
|
|
|
|
res.ip
|
|
|
|
) {
|
|
|
|
if (isPrivateIp(res.ip)) {
|
2024-04-25 04:08:45 +02:00
|
|
|
downloadLogger.warn(`Blocked address: ${res.ip}`);
|
2023-01-13 05:40:33 +01:00
|
|
|
req.destroy();
|
|
|
|
}
|
2021-09-03 14:00:44 +02:00
|
|
|
}
|
2021-09-04 13:33:14 +02:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
const contentLength = res.headers["content-length"];
|
|
|
|
if (contentLength != null) {
|
|
|
|
const size = Number(contentLength);
|
|
|
|
if (size > maxSize) {
|
2024-04-25 04:08:45 +02:00
|
|
|
downloadLogger.warn(
|
|
|
|
`maxSize exceeded (${size} > ${maxSize}) on response`,
|
|
|
|
);
|
2023-01-13 05:40:33 +01:00
|
|
|
req.destroy();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.on("downloadProgress", (progress: Got.Progress) => {
|
|
|
|
if (progress.transferred > maxSize) {
|
2024-04-25 04:08:45 +02:00
|
|
|
downloadLogger.warn(
|
2023-01-13 05:40:33 +01:00
|
|
|
`maxSize exceeded (${progress.transferred} > ${maxSize}) on downloadProgress`,
|
|
|
|
);
|
2021-09-04 13:33:14 +02:00
|
|
|
req.destroy();
|
|
|
|
}
|
2023-01-13 05:40:33 +01:00
|
|
|
});
|
2019-03-20 20:50:44 +01:00
|
|
|
|
2021-10-16 10:16:24 +02:00
|
|
|
try {
|
2024-04-23 22:19:03 +02:00
|
|
|
await stream.pipeline(req, fs.createWriteStream(path));
|
2021-10-16 10:16:24 +02:00
|
|
|
} catch (e) {
|
|
|
|
if (e instanceof Got.HTTPError) {
|
2023-01-13 05:40:33 +01:00
|
|
|
throw new StatusError(
|
|
|
|
`${e.response.statusCode} ${e.response.statusMessage}`,
|
|
|
|
e.response.statusCode,
|
|
|
|
e.response.statusMessage,
|
|
|
|
);
|
2021-10-16 10:16:24 +02:00
|
|
|
} else {
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
}
|
2019-03-20 20:50:44 +01:00
|
|
|
|
2024-04-25 04:08:45 +02:00
|
|
|
downloadLogger.debug(`Download finished: ${chalk.cyan(url)}`);
|
2019-03-20 20:50:44 +01:00
|
|
|
}
|
2021-09-03 14:00:44 +02:00
|
|
|
|
2023-08-18 11:51:41 +02:00
|
|
|
export function isPrivateIp(ip: string): boolean {
|
2021-09-03 14:00:44 +02:00
|
|
|
for (const net of config.allowedPrivateNetworks || []) {
|
|
|
|
const cidr = new IPCIDR(net);
|
|
|
|
if (cidr.contains(ip)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return PrivateIp(ip);
|
|
|
|
}
|