modded7/buildtools/util/curseForgeAPI.ts
2022-07-07 16:11:30 +08:00

212 lines
5.9 KiB
TypeScript

import bluebird from "bluebird";
import { CurseForgeFetchedFileInfo, CurseForgeModInfo as CurseForgeProject } from "../types/curseForge";
import log from "fancy-log";
import request from "requestretry";
import { ModpackManifestFile } from "../types/modpackManifest";
import Bluebird from "bluebird";
import buildConfig from "../buildConfig";
import upath from "upath";
import fs from "fs";
import { FileDef } from "../types/fileDef";
import { downloadOrRetrieveFileDef, relative, RetrievedFileDefReason } from "./util";
function getCurseForgeToken() {
const vari = "CFCORE_API_TOKEN";
const val = process.env[vari];
if (!process.env[vari]) {
throw new Error(`Environmental variable ${vari} is unset.`);
}
return val;
}
const curseForgeProjectCache: { [key: number]: CurseForgeProject } = {};
export async function fetchProject(toFetch: number): Promise<CurseForgeProject> {
if (curseForgeProjectCache[toFetch]) {
return curseForgeProjectCache[toFetch];
}
const project: CurseForgeProject | undefined = (
await request({
uri: `${buildConfig.cfCoreApiEndpoint}/v1/mods/${toFetch}`,
json: true,
fullResponse: false,
maxAttempts: 5,
headers: {
"X-Api-Key": getCurseForgeToken(),
},
})
)?.data;
if (!project) {
throw new Error(`Failed to fetch project ${toFetch}`);
}
if (project) {
curseForgeProjectCache[toFetch] = project;
}
return project;
}
const fetchedFileInfoCache: { [key: string]: CurseForgeFetchedFileInfo } = {};
export async function fetchFileInfo(projectID: number, fileID: number): Promise<CurseForgeFetchedFileInfo> {
const slug = `${projectID}/${fileID}`;
if (fetchedFileInfoCache[slug]) {
return fetchedFileInfoCache[slug];
}
const fileInfo: CurseForgeFetchedFileInfo = (
await request({
uri: `${buildConfig.cfCoreApiEndpoint}/v1/mods/${projectID}/files/${fileID}`,
json: true,
fullResponse: false,
headers: {
"X-Api-Key": getCurseForgeToken(),
},
})
)?.data;
if (!fileInfo) {
throw new Error(`Failed to download file ${projectID}/file/${fileID}`);
}
if (fileInfo) {
fetchedFileInfoCache[slug] = fileInfo;
if (!fileInfo.downloadUrl) {
const fid = `${Math.floor(fileInfo.id / 1000)}/${fileInfo.id % 1000}`;
fileInfo.downloadUrl = `https://edge.forgecdn.net/files/${fid}/${fileInfo.fileName}`;
}
}
return fileInfo;
}
/**
* Fetches multiple CurseForge projects.
* Falls back to fetchModInfo in case it's impossible to bulk-fetch some projects.
*
* @param toFetch Project IDs to fetch.
* @returns CurseForge project infos.
*/
export async function fetchProjectsBulk(toFetch: number[]): Promise<CurseForgeProject[]> {
const modInfos: CurseForgeProject[] = [];
const unfetched: number[] = [];
// Determine projects that have been fetched already.
toFetch.forEach((id) => {
const cached = curseForgeProjectCache[id];
if (cached) {
modInfos.push(cached);
} else {
unfetched.push(id);
}
});
if (unfetched.length > 0) {
// Augment the array of known projects with new info.
const fetched: CurseForgeProject[] = (
await request.post({
uri: `${buildConfig.cfCoreApiEndpoint}/v1/mods`,
json: {
modIds: unfetched,
},
fullResponse: false,
maxAttempts: 5,
headers: {
"X-Api-Key": getCurseForgeToken(),
},
})
)?.data;
if (!fetched) {
throw new Error(`Failed to bulk-fetch projects ${unfetched.join(", ")}`);
}
modInfos.push(...fetched);
// Cache fetched stuff.
fetched.forEach((mi) => {
curseForgeProjectCache[mi.id] = mi;
});
// In case we haven't received the proper amount of mod infos,
// try requesting them individually.
if (unfetched.length !== toFetch.length) {
const modInfoIDs = new Set(modInfos.map((mi) => mi.id));
const toFetchMissing = [...new Set(toFetch.filter((x) => !modInfoIDs.has(x)))];
log.warn(`Couldn't fetch next project IDs in bulk: ${toFetchMissing.join(", ")}`);
// Try fetching mods individually, in case they've been deleted.
let count = 0;
const missingModInfos: CurseForgeProject[] = await bluebird.map(toFetchMissing, async (id) => {
log.info(`Fetching project ID ${id} directly... (${++count} / ${toFetchMissing.length})`);
try {
// In case something fails to download; catch, rewrite, rethrow.
return await fetchProject(id);
} catch (err) {
err.message = `Couldn't fetch project ID ${id}. ${err.message || "Unknown error"}`;
throw err;
}
});
// The code above is expected to throw and terminate the further execution,
// so we can just do this.
modInfos.push(...missingModInfos);
}
}
return modInfos;
}
export async function fetchMods(toFetch: ModpackManifestFile[], destination: string): Promise<void[]> {
if (toFetch.length > 0) {
log(`Fetching ${toFetch.length} mods...`);
const modsPath = upath.join(destination, "mods");
await fs.promises.mkdir(modsPath, { recursive: true });
let fetched = 0;
return Bluebird.map(
toFetch,
async (file) => {
const fileInfo = await fetchFileInfo(file.projectID, file.fileID);
const fileDef: FileDef = {
url: fileInfo.downloadUrl,
};
// https://docs.curseforge.com/#tocS_GetModsByIdsListRequestBody
if (fileInfo.hashes) {
fileDef.hashes = fileInfo.hashes.map((hash) => ({
hashes: hash.value,
id: hash.algo == 1 ? "sha1" : "md5",
}));
}
const modFile = await downloadOrRetrieveFileDef(fileDef);
fetched += 1;
if (modFile.reason == RetrievedFileDefReason.Downloaded) {
log(`Downloaded ${upath.basename(fileDef.url)}... (${fetched} / ${toFetch.length})`);
} else if (modFile.reason == RetrievedFileDefReason.CacheHit) {
log(`Fetched ${upath.basename(fileDef.url)} from cache... (${fetched} / ${toFetch.length})`);
}
const dest = upath.join(destination, "mods", fileInfo.fileName);
await fs.promises.symlink(relative(dest, modFile.cachePath), dest);
},
{ concurrency: buildConfig.downloaderConcurrency },
);
} else {
log("No mods to fetch.");
}
}