mirror of
https://github.com/thepeacockproject/Peacock
synced 2024-11-22 22:12:45 +01:00
260 lines
7.8 KiB
TypeScript
260 lines
7.8 KiB
TypeScript
/*
|
|
* The Peacock Project - a HITMAN server replacement.
|
|
* Copyright (C) 2021-2023 The Peacock Project Team
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Affero General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
import prompts from "prompts"
|
|
import { log, LogLevel } from "./loggingInterop"
|
|
import { readdir, writeFile } from "fs/promises"
|
|
import { PEACOCKVER, PEACOCKVERSTRING } from "./utils"
|
|
import { getSwizzleable } from "./configSwizzleManager"
|
|
import md5File from "md5-file"
|
|
import { arch, cpus as cpuList, platform, version } from "os"
|
|
import { Controller, isPlugin } from "./controller"
|
|
import { getAllFlags } from "./flags"
|
|
import axios from "axios"
|
|
import { Stream } from "stream"
|
|
import { createWriteStream, existsSync, mkdirSync } from "fs"
|
|
import ProgressBar from "progress"
|
|
import { resolve as pathResolve } from "path"
|
|
import picocolors from "picocolors"
|
|
import { Filename, npath, PortablePath, ppath, xfs } from "@yarnpkg/fslib"
|
|
import { makeEmptyArchive, ZipFS } from "@yarnpkg/libzip"
|
|
|
|
const IMAGE_PACK_REPO = "thepeacockproject/ImagePack"
|
|
|
|
export async function toolsMenu() {
|
|
const init = await prompts({
|
|
name: "actions",
|
|
message: "Select actions:",
|
|
type: "select",
|
|
choices: [
|
|
{
|
|
title: "Export debug info",
|
|
description: "Export helpful information for the developers.",
|
|
value: "debug",
|
|
},
|
|
{
|
|
title: "Download contract from IOI servers",
|
|
description: "Download a contract from IOI's servers.",
|
|
value: "download-contract",
|
|
},
|
|
{
|
|
title: "Download all assets for offline use",
|
|
description:
|
|
"Downloads all the files you need to use Peacock fully offline.",
|
|
value: "download-images",
|
|
},
|
|
],
|
|
})
|
|
|
|
switch (init.actions) {
|
|
case "debug":
|
|
await exportDebugInfo()
|
|
break
|
|
case "download-images":
|
|
await downloadImagePack()
|
|
break
|
|
case "download-contract":
|
|
await downloadContract()
|
|
break
|
|
default:
|
|
log(LogLevel.ERROR, "Unknown action!")
|
|
}
|
|
}
|
|
|
|
async function copyIntoZip(zip: ZipFS, path: string): Promise<void> {
|
|
await zip.copyPromise(zip.resolve(path as Filename), ppath.resolve(path), {
|
|
stableTime: true,
|
|
stableSort: true,
|
|
baseFs: xfs,
|
|
})
|
|
}
|
|
|
|
async function exportDebugInfo(): Promise<void> {
|
|
const cpus = cpuList().map((cpu, index) => ({
|
|
core: index + 1,
|
|
...cpu,
|
|
}))
|
|
|
|
const files = [
|
|
...(await readdir(process.cwd())),
|
|
...(await readdir(pathResolve(process.cwd(), "plugins"))).map(
|
|
(file) => `plugins/${file}`,
|
|
),
|
|
]
|
|
const plugins = await Promise.all(
|
|
[
|
|
...files.filter((file) => isPlugin(file, "js")),
|
|
...files.filter((file) => isPlugin(file, "cjs")),
|
|
].map(async (plugin) => {
|
|
return `${plugin} (${await md5File(plugin)})`
|
|
}),
|
|
)
|
|
|
|
const data = {
|
|
version: PEACOCKVERSTRING,
|
|
ident: PEACOCKVER,
|
|
presentConfigs: getSwizzleable(),
|
|
chunkDigest: await md5File("chunk0.js"),
|
|
patcherDigest: await md5File("PeacockPatcher.exe"),
|
|
runtimeVersions: process.versions,
|
|
os: `${platform()} (${arch()}) - Release: ${version()}`,
|
|
cpus,
|
|
plugins,
|
|
flags: getAllFlags(),
|
|
}
|
|
|
|
const debugJson = JSON.stringify(data, undefined, 4)
|
|
|
|
const zipFile = ppath.resolve(ppath.cwd(), "DEBUG_PROFILE.zip")
|
|
|
|
// we'll start by creating an empty zip file
|
|
await writeFile(npath.fromPortablePath(zipFile), makeEmptyArchive())
|
|
|
|
const zip = new ZipFS(zipFile, { create: true })
|
|
|
|
await zip.writeFilePromise(zip.resolve("meta.json" as Filename), debugJson)
|
|
|
|
await copyIntoZip(zip, "logs")
|
|
await copyIntoZip(zip, "userdata")
|
|
await copyIntoZip(zip, "contractSessions")
|
|
await copyIntoZip(zip, "contracts")
|
|
|
|
zip.saveAndClose()
|
|
|
|
log(
|
|
LogLevel.INFO,
|
|
"Successfully outputted debugging data to DEBUG_PROFILE.zip!",
|
|
)
|
|
}
|
|
|
|
async function downloadContract(): Promise<void> {
|
|
log(LogLevel.INFO, "Contract downloading tool - powered by HITMAPS")
|
|
log(
|
|
LogLevel.INFO,
|
|
"NOTE: This tool only works for HITMAN 3 contracts that are on Stadia, Steam, Epic, or PlayStation.",
|
|
)
|
|
|
|
const { contractId } = await prompts({
|
|
type: "text",
|
|
name: "contractId",
|
|
message: "Enter the contract ID (with dashes)",
|
|
})
|
|
|
|
const result = await Controller._hitmapsFetchContract(contractId)
|
|
|
|
if (!result) {
|
|
log(
|
|
LogLevel.ERROR,
|
|
`Unable to resolve ${contractId}. This may be because HITMAPS' bot is not authenticated, or the contract was not found.`,
|
|
)
|
|
return
|
|
}
|
|
|
|
result.Metadata.CreatorUserId = "fadb923c-e6bb-4283-a537-eb4d1150262e"
|
|
|
|
if (!existsSync("contracts")) {
|
|
mkdirSync("contracts")
|
|
}
|
|
|
|
await writeFile(
|
|
`contracts/${contractId}.json`,
|
|
JSON.stringify(result, undefined, 4),
|
|
)
|
|
|
|
log(LogLevel.INFO, "Successfully saved the contract!")
|
|
}
|
|
|
|
async function downloadImagePack(): Promise<void> {
|
|
// the code below very likely triggers a memory leak!
|
|
// the streams might not actually be destroyed properly, but I don't really care, as the process
|
|
// should end after running this
|
|
|
|
const writer = createWriteStream(
|
|
pathResolve(__dirname, "offlineassets.zip"),
|
|
)
|
|
|
|
log(LogLevel.INFO, "Starting asset download...")
|
|
|
|
let resp, totalLength
|
|
|
|
try {
|
|
const releaseInfo = await axios.get(
|
|
`https://api.github.com/repos/${IMAGE_PACK_REPO}/releases/latest`,
|
|
)
|
|
|
|
if (releaseInfo.status !== 200) {
|
|
throw new Error("Failed to get release info!")
|
|
}
|
|
|
|
// eslint-disable-next-line prefer-const
|
|
totalLength = releaseInfo.data["assets"][0]["size"]
|
|
|
|
// eslint-disable-next-line prefer-const
|
|
resp = await axios.get<Stream>(
|
|
releaseInfo.data["assets"][0]["browser_download_url"],
|
|
{
|
|
responseType: "stream",
|
|
},
|
|
)
|
|
} catch (e) {
|
|
log(LogLevel.ERROR, "Unable to complete download due to an error!")
|
|
throw e
|
|
}
|
|
|
|
const progressBar = new ProgressBar(
|
|
`${picocolors.blue("--> Downloading")} [:bar] ${picocolors.magenta(
|
|
"(:percent)",
|
|
)}`,
|
|
{
|
|
width: 40,
|
|
complete: "=",
|
|
incomplete: " ",
|
|
renderThrottle: 1,
|
|
total: totalLength,
|
|
},
|
|
)
|
|
|
|
resp.data.on("data", (chunk) => {
|
|
progressBar.tick(chunk.length)
|
|
})
|
|
|
|
resp.data.pipe(writer)
|
|
|
|
await new Promise<void>((resolve) => {
|
|
writer.on("finish", resolve)
|
|
})
|
|
|
|
log(LogLevel.INFO, "Extracting files...")
|
|
|
|
const zipFS = new ZipFS(ppath.resolve("offlineassets.zip" as Filename), {
|
|
readOnly: true,
|
|
})
|
|
|
|
await xfs.copyPromise(
|
|
ppath.resolve("images" as PortablePath),
|
|
`/images` as PortablePath,
|
|
{
|
|
baseFs: zipFS,
|
|
overwrite: true,
|
|
},
|
|
)
|
|
|
|
log(LogLevel.INFO, "Done!")
|
|
await xfs.unlinkPromise("offlineassets.zip" as Filename)
|
|
}
|