mod-manager/src/mods/mods.ts
2022-08-07 20:09:42 +01:00

241 lines
8.9 KiB
TypeScript

import path from "path";
import PrintUtils from "../util/print_utils.js";
import ModrinthSource from "./sources/modrinth_source.js";
import ModSource from "./sources/mod_source.js";
import ModNotFoundError from "../errors/mod_not_found_error.js";
import {readFileSync, unlinkSync, writeFileSync} from "fs";
import Util from "../util/util.js";
import ModManager from "../mod-manager.js";
import MinecraftUtils from "../util/minecraft_utils.js";
export default class Mods {
private static readonly MOD_SOURCES: Array<ModSource> = [
new ModrinthSource()
];
public static async install(mod: string, essential: boolean): Promise<void> {
let success: boolean = false;
// Go through each mod source
for (const source of this.MOD_SOURCES) {
// If we have not yet successfully installed the queried mod
if (!success) {
const spinner = new PrintUtils.Spinner(`Searching for ${mod}...`);
spinner.start();
// Search for the mod
let id: string | undefined;
try {
id = await source.search(mod);
} catch (e) {
if (e instanceof ModNotFoundError) {
spinner.stop(`Mod not found on ${source.getSourceName()}`)
} else {
spinner.error(`An error occurred searching for ${mod} on ${source.getSourceName()}. Skipping ${source.getSourceName()}`)
// Try the next source
continue;
}
}
// If a mod is found, install it
if (id != undefined) {
const projectName = await source.getProjectName(id);
// If mod is not already installed
if (!this.isModInstalled(id)) {
spinner.updateText(`Installing ${projectName}...`)
try {
const mcVersion = await MinecraftUtils.getCurrentMinecraftVersion();
const latestVersion = await source.getLatestVersion(id, mcVersion)
const modObj: Mod = await source.install(latestVersion, essential);
this.trackMod(modObj);
spinner.succeed(`Successfully installed ${projectName}`);
success = true;
} catch (e) {
// Log the error, and continue to next source
spinner.error(e);
}
} else {
spinner.error(`Mod ${projectName} is already installed.`)
}
}
}
}
}
private static trackMod(mod: Mod): void {
// Read current file
const mods = this.getTrackedMods();
// Add mod
mods.push(mod);
// Write list back to file
this.writeFile(mods);
}
public static getTrackedMods(): Array<Mod> {
const file = readFileSync(ModManager.FilePaths.MOD_FILE_PATH, "utf-8");
return JSON.parse(file);
}
public static writeFile(mods: Array<Mod>): void {
writeFileSync(ModManager.FilePaths.MOD_FILE_PATH, JSON.stringify(mods, null, 4));
}
private static isModInstalled(id: string): boolean {
const modsWithId: Array<Mod> = this.getTrackedMods().filter(mod => mod.id == id);
return !Util.isArrayEmpty(modsWithId)
}
static uninstall(mod: string) {
// Find mod to uninstall
const spinner = new PrintUtils.Spinner(`Uninstalling ${mod}...`)
spinner.start();
const modToUninstall = this.findMod(mod);
// IF a matching mod is found, remove it
if (modToUninstall != undefined) {
this.silentUninstall(modToUninstall);
spinner.succeed(`${modToUninstall.name} successfully uninstalled!`)
} else {
spinner.error(`${mod} was not found.`)
}
}
static silentUninstall(mod: Mod) {
let mods: Array<Mod> = this.getTrackedMods();
// Remove mod from list and uninstall it
unlinkSync(path.join(ModManager.FilePaths.MODS_FOLDER_PATH, mod.fileName));
mods = mods.filter(item => !Mods.areModsEqual(item, mod));
this.writeFile(mods);
}
static areModsEqual(mod1: Mod, mod2: Mod): boolean {
return mod1.id === mod2.id;
}
static toggleEssential(mod: string) {
const modToMark = this.findMod(mod);
if (modToMark != undefined) {
let mods = this.getTrackedMods();
// Remove mod from list
mods = mods.filter(item => !Mods.areModsEqual(item, modToMark));
// Toggle essnetial status, and write back to file
modToMark.essential = !modToMark.essential;
mods.push(modToMark)
this.writeFile(mods);
if (modToMark.essential) {
PrintUtils.success(`Marked ${modToMark.name} as essential`)
} else {
PrintUtils.success(`Marked ${modToMark.name} as inessential`)
}
} else {
PrintUtils.error(`${mod} not found.`)
}
}
private static findMod(mod: string): Mod | undefined {
// Replace underscores with spaces
mod = mod.replaceAll("_", " ");
let mods: Array<Mod> = this.getTrackedMods();
for (let modEle of mods) {
const id = modEle.id.toLowerCase();
const name = modEle.name.toLowerCase();
const query = mod.toLowerCase();
if (id == query || Util.areStringsSimilar(mod, name)) {
return modEle;
}
}
return undefined;
}
static async update() {
const trackedMods = this.getTrackedMods();
if (Util.isArrayEmpty(trackedMods)) {
PrintUtils.error("There are no mods currently installed. Try `mod-manager install -h` to learn more!")
return;
}
const mcVersion = await MinecraftUtils.getCurrentMinecraftVersion();
// For every tracked mod
for (let mod of trackedMods) {
const spinner = new PrintUtils.Spinner(`Checking for newer version of ${mod.name}`)
spinner.start();
// Get the latest version
const source = this.getSourceFromName(mod.source);
let latestVersion: Version | undefined = undefined;
try {
latestVersion = await source.getLatestVersion(mod.id, mcVersion);
// If the latest version has a different version number, it must be newer, install it.
if (latestVersion.version_number != mod.version) {
spinner.updateText(`Newer version for ${mod.name} found. Installing...`)
this.silentUninstall(mod);
const newMod = await source.install(latestVersion, mod.essential);
this.trackMod(newMod);
spinner.succeed(`Successfully updated ${newMod.name}`)
// Else, the latest version is already installed, do nothing.
} else {
throw new ModNotFoundError("There is no newer version available.");
}
} catch (e) {
spinner.error(`${mod.name} already has the latest version installed!`)
}
}
}
static getSourceFromName(name: string): ModSource {
const source = this.MOD_SOURCES.filter(src => src.getSourceName() === name)[0];
if (source == undefined) {
throw new Error(`There is no source registered with the name ${name}`)
}
return source
}
static async isMigratePossible(version: string): Promise<boolean> {
const mods = this.getTrackedMods();
let availableList = [];
// For every tracked mod
for (let mod of mods) {
// Get the latest version for each mod on the provided minecraft version
const spinner = new PrintUtils.Spinner(`Checking ${mod.name}...`);
spinner.start();
const source = this.getSourceFromName(mod.source);
try {
await source.getLatestVersion(mod.id, version)
// Report and record that this mod is available
spinner.succeed(`${mod.name} is available on Minecraft ${version}`)
availableList.push(true)
} catch (e) {
// Report and record that this mod is not available
spinner.error(`${mod.name} is not available on Minecraft ${version}`)
availableList.push(false);
}
}
// Filter out all the true's from the list
availableList = availableList.filter(available => !available)
// If the array is empty, all the mods reported as available, and a migration is possible
return Util.isArrayEmpty(availableList);
}
}