2025-03-20 05:36:09 -07:00
|
|
|
import { getJSONfromString } from "@/src/helpers/stringHelpers";
|
2025-03-20 15:27:37 -07:00
|
|
|
import { freeUpSlot, getInventory } from "@/src/services/inventoryService";
|
2025-03-20 05:36:09 -07:00
|
|
|
import { getAccountIdForRequest } from "@/src/services/loginService";
|
|
|
|
import { SRng } from "@/src/services/rngService";
|
2025-03-20 15:27:37 -07:00
|
|
|
import { IMongoDate, IOid } from "@/src/types/commonTypes";
|
|
|
|
import {
|
|
|
|
IInfNode,
|
|
|
|
IInnateDamageFingerprint,
|
|
|
|
InventorySlot,
|
|
|
|
TEquipmentKey
|
|
|
|
} from "@/src/types/inventoryTypes/inventoryTypes";
|
2025-03-20 05:36:09 -07:00
|
|
|
import { logger } from "@/src/utils/logger";
|
|
|
|
import { RequestHandler } from "express";
|
|
|
|
import { ExportRegions } from "warframe-public-export-plus";
|
|
|
|
|
|
|
|
export const nemesisController: RequestHandler = async (req, res) => {
|
2025-03-20 15:27:37 -07:00
|
|
|
const accountId = await getAccountIdForRequest(req);
|
|
|
|
if ((req.query.mode as string) == "f") {
|
|
|
|
const body = getJSONfromString<IValenceFusionRequest>(String(req.body));
|
|
|
|
const inventory = await getInventory(accountId, body.Category + " WeaponBin");
|
|
|
|
const destWeapon = inventory[body.Category].id(body.DestWeapon.$oid)!;
|
|
|
|
const sourceWeapon = inventory[body.Category].id(body.SourceWeapon.$oid)!;
|
|
|
|
const destFingerprint = JSON.parse(destWeapon.UpgradeFingerprint!) as IInnateDamageFingerprint;
|
|
|
|
const sourceFingerprint = JSON.parse(sourceWeapon.UpgradeFingerprint!) as IInnateDamageFingerprint;
|
|
|
|
|
|
|
|
// Upgrade destination damage type if desireed
|
|
|
|
if (body.UseSourceDmgType) {
|
|
|
|
destFingerprint.buffs[0].Tag = sourceFingerprint.buffs[0].Tag;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Upgrade destination damage value
|
|
|
|
const destDamage = 0.25 + (destFingerprint.buffs[0].Value / 0x3fffffff) * (0.6 - 0.25);
|
|
|
|
const sourceDamage = 0.25 + (sourceFingerprint.buffs[0].Value / 0x3fffffff) * (0.6 - 0.25);
|
|
|
|
let newDamage = Math.max(destDamage, sourceDamage) * 1.1;
|
|
|
|
if (newDamage >= 0.58) {
|
|
|
|
newDamage = 0.6;
|
|
|
|
}
|
|
|
|
destFingerprint.buffs[0].Value = Math.trunc(((newDamage - 0.25) / (0.6 - 0.25)) * 0x3fffffff);
|
|
|
|
|
|
|
|
// Commit fingerprint
|
|
|
|
destWeapon.UpgradeFingerprint = JSON.stringify(destFingerprint);
|
|
|
|
|
|
|
|
// Remove source weapon
|
|
|
|
inventory[body.Category].pull({ _id: body.SourceWeapon.$oid });
|
|
|
|
freeUpSlot(inventory, InventorySlot.WEAPONS);
|
|
|
|
|
|
|
|
await inventory.save();
|
|
|
|
res.json({
|
|
|
|
InventoryChanges: {
|
|
|
|
[body.Category]: [destWeapon.toJSON()]
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else if ((req.query.mode as string) == "s") {
|
2025-03-20 05:36:09 -07:00
|
|
|
const inventory = await getInventory(accountId, "Nemesis NemesisAbandonedRewards");
|
|
|
|
const body = getJSONfromString<INemesisStartRequest>(String(req.body));
|
2025-03-20 15:27:15 -07:00
|
|
|
body.target.fp = BigInt(body.target.fp);
|
|
|
|
|
|
|
|
let infNodes: IInfNode[];
|
|
|
|
let weaponIdx = -1;
|
|
|
|
if (body.target.Faction == "FC_INFESTATION") {
|
|
|
|
infNodes = [
|
|
|
|
{
|
|
|
|
Node: "SolNode852",
|
|
|
|
Influence: 1
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Node: "SolNode850",
|
|
|
|
Influence: 1
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Node: "SolNode851",
|
|
|
|
Influence: 1
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Node: "SolNode853",
|
|
|
|
Influence: 1
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Node: "SolNode854",
|
|
|
|
Influence: 1
|
|
|
|
}
|
|
|
|
];
|
|
|
|
} else {
|
|
|
|
infNodes = [];
|
|
|
|
for (const [key, value] of Object.entries(ExportRegions)) {
|
|
|
|
if (
|
|
|
|
value.systemIndex == 2 && // earth
|
|
|
|
value.nodeType != 3 && // not hub
|
|
|
|
value.nodeType != 7 && // not junction
|
|
|
|
value.missionIndex && // must have a mission type and not assassination
|
|
|
|
value.missionIndex != 28 && // not open world
|
|
|
|
value.missionIndex != 32 && // not railjack
|
|
|
|
value.missionIndex != 41 && // not saya's visions
|
|
|
|
value.name.indexOf("Archwing") == -1
|
|
|
|
) {
|
|
|
|
//console.log(dict_en[value.name]);
|
|
|
|
infNodes.push({ Node: key, Influence: 1 });
|
|
|
|
}
|
|
|
|
}
|
2025-03-20 05:36:09 -07:00
|
|
|
|
2025-03-20 15:27:15 -07:00
|
|
|
let weapons: readonly string[];
|
|
|
|
if (body.target.manifest == "/Lotus/Types/Game/Nemesis/KuvaLich/KuvaLichManifestVersionSix") {
|
|
|
|
weapons = kuvaLichVersionSixWeapons;
|
|
|
|
} else if (
|
|
|
|
body.target.manifest == "/Lotus/Types/Enemies/Corpus/Lawyers/LawyerManifestVersionFour" ||
|
|
|
|
body.target.manifest == "/Lotus/Types/Enemies/Corpus/Lawyers/LawyerManifestVersionThree"
|
2025-03-20 05:36:09 -07:00
|
|
|
) {
|
2025-03-20 15:27:15 -07:00
|
|
|
weapons = corpusVersionThreeWeapons;
|
|
|
|
} else {
|
|
|
|
throw new Error(`unknown nemesis manifest: ${body.target.manifest}`);
|
2025-03-20 05:36:09 -07:00
|
|
|
}
|
|
|
|
|
2025-03-20 15:27:15 -07:00
|
|
|
const initialWeaponIdx = new SRng(body.target.fp).randomInt(0, weapons.length - 1);
|
|
|
|
weaponIdx = initialWeaponIdx;
|
|
|
|
do {
|
|
|
|
const weapon = weapons[weaponIdx];
|
|
|
|
if (!body.target.DisallowedWeapons.find(x => x == weapon)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
weaponIdx = (weaponIdx + 1) % weapons.length;
|
|
|
|
} while (weaponIdx != initialWeaponIdx);
|
2025-03-20 05:36:09 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
inventory.Nemesis = {
|
|
|
|
fp: body.target.fp,
|
|
|
|
manifest: body.target.manifest,
|
|
|
|
KillingSuit: body.target.KillingSuit,
|
|
|
|
killingDamageType: body.target.killingDamageType,
|
|
|
|
ShoulderHelmet: body.target.ShoulderHelmet,
|
|
|
|
WeaponIdx: weaponIdx,
|
|
|
|
AgentIdx: body.target.AgentIdx,
|
|
|
|
BirthNode: body.target.BirthNode,
|
|
|
|
Faction: body.target.Faction,
|
|
|
|
Rank: 0,
|
|
|
|
k: false,
|
|
|
|
Traded: false,
|
|
|
|
d: new Date(),
|
|
|
|
InfNodes: infNodes,
|
|
|
|
GuessHistory: [],
|
|
|
|
Hints: [],
|
|
|
|
HintProgress: 0,
|
|
|
|
Weakened: body.target.Weakened,
|
|
|
|
PrevOwners: 0,
|
|
|
|
HenchmenKilled: 0,
|
|
|
|
SecondInCommand: body.target.SecondInCommand
|
|
|
|
};
|
|
|
|
inventory.NemesisAbandonedRewards = []; // unclear if we need to do this since the client also submits this with missionInventoryUpdate
|
|
|
|
await inventory.save();
|
|
|
|
|
|
|
|
res.json({
|
|
|
|
target: inventory.toJSON().Nemesis
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
logger.debug(`data provided to ${req.path}: ${String(req.body)}`);
|
|
|
|
throw new Error(`unknown nemesis mode: ${String(req.query.mode)}`);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2025-03-20 15:27:37 -07:00
|
|
|
interface IValenceFusionRequest {
|
|
|
|
DestWeapon: IOid;
|
|
|
|
SourceWeapon: IOid;
|
|
|
|
Category: TEquipmentKey;
|
|
|
|
UseSourceDmgType: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface INemesisStartRequest {
|
2025-03-20 05:36:09 -07:00
|
|
|
target: {
|
|
|
|
fp: number | bigint;
|
|
|
|
manifest: string;
|
|
|
|
KillingSuit: string;
|
|
|
|
killingDamageType: number;
|
|
|
|
ShoulderHelmet: string;
|
|
|
|
DisallowedWeapons: string[];
|
|
|
|
WeaponIdx: number;
|
|
|
|
AgentIdx: number;
|
|
|
|
BirthNode: string;
|
|
|
|
Faction: string;
|
|
|
|
Rank: number;
|
|
|
|
k: boolean;
|
|
|
|
Traded: boolean;
|
|
|
|
d: IMongoDate;
|
|
|
|
InfNodes: [];
|
|
|
|
GuessHistory: [];
|
|
|
|
Hints: [];
|
|
|
|
HintProgress: number;
|
|
|
|
Weakened: boolean;
|
|
|
|
PrevOwners: number;
|
|
|
|
HenchmenKilled: number;
|
2025-03-20 15:27:15 -07:00
|
|
|
MissionCount?: number; // Added in 38.5.0
|
|
|
|
LastEnc?: number; // Added in 38.5.0
|
2025-03-20 05:36:09 -07:00
|
|
|
SecondInCommand: boolean;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
const kuvaLichVersionSixWeapons = [
|
|
|
|
"/Lotus/Weapons/Grineer/KuvaLich/LongGuns/Drakgoon/KuvaDrakgoon",
|
|
|
|
"/Lotus/Weapons/Grineer/KuvaLich/LongGuns/Karak/KuvaKarak",
|
|
|
|
"/Lotus/Weapons/Grineer/Melee/GrnKuvaLichScythe/GrnKuvaLichScytheWeapon",
|
|
|
|
"/Lotus/Weapons/Grineer/KuvaLich/LongGuns/Kohm/KuvaKohm",
|
|
|
|
"/Lotus/Weapons/Grineer/KuvaLich/LongGuns/Ogris/KuvaOgris",
|
|
|
|
"/Lotus/Weapons/Grineer/KuvaLich/LongGuns/Quartakk/KuvaQuartakk",
|
|
|
|
"/Lotus/Weapons/Grineer/KuvaLich/LongGuns/Tonkor/KuvaTonkor",
|
|
|
|
"/Lotus/Weapons/Grineer/KuvaLich/Secondaries/Brakk/KuvaBrakk",
|
|
|
|
"/Lotus/Weapons/Grineer/KuvaLich/Secondaries/Kraken/KuvaKraken",
|
|
|
|
"/Lotus/Weapons/Grineer/KuvaLich/Secondaries/Seer/KuvaSeer",
|
|
|
|
"/Lotus/Weapons/Grineer/KuvaLich/Secondaries/Stubba/KuvaStubba",
|
|
|
|
"/Lotus/Weapons/Grineer/HeavyWeapons/GrnHeavyGrenadeLauncher",
|
|
|
|
"/Lotus/Weapons/Grineer/LongGuns/GrnKuvaLichRifle/GrnKuvaLichRifleWeapon",
|
|
|
|
"/Lotus/Weapons/Grineer/Bows/GrnBow/GrnBowWeapon",
|
|
|
|
"/Lotus/Weapons/Grineer/KuvaLich/LongGuns/Hind/KuvaHind",
|
|
|
|
"/Lotus/Weapons/Grineer/KuvaLich/Secondaries/Nukor/KuvaNukor",
|
|
|
|
"/Lotus/Weapons/Grineer/KuvaLich/LongGuns/Hek/KuvaHekWeapon",
|
|
|
|
"/Lotus/Weapons/Grineer/KuvaLich/LongGuns/Zarr/KuvaZarr",
|
|
|
|
"/Lotus/Weapons/Grineer/KuvaLich/HeavyWeapons/Grattler/KuvaGrattler",
|
|
|
|
"/Lotus/Weapons/Grineer/KuvaLich/LongGuns/Sobek/KuvaSobek"
|
|
|
|
];
|
|
|
|
|
|
|
|
const corpusVersionThreeWeapons = [
|
|
|
|
"/Lotus/Weapons/Corpus/LongGuns/CrpBriefcaseLauncher/CrpBriefcaseLauncher",
|
|
|
|
"/Lotus/Weapons/Corpus/BoardExec/Primary/CrpBEArcaPlasmor/CrpBEArcaPlasmor",
|
|
|
|
"/Lotus/Weapons/Corpus/BoardExec/Primary/CrpBEFluxRifle/CrpBEFluxRifle",
|
|
|
|
"/Lotus/Weapons/Corpus/BoardExec/Primary/CrpBETetra/CrpBETetra",
|
|
|
|
"/Lotus/Weapons/Corpus/BoardExec/Secondary/CrpBECycron/CrpBECycron",
|
|
|
|
"/Lotus/Weapons/Corpus/BoardExec/Secondary/CrpBEDetron/CrpBEDetron",
|
|
|
|
"/Lotus/Weapons/Corpus/Pistols/CrpIgniterPistol/CrpIgniterPistol",
|
|
|
|
"/Lotus/Weapons/Corpus/Pistols/CrpBriefcaseAkimbo/CrpBriefcaseAkimboPistol",
|
|
|
|
"/Lotus/Weapons/Corpus/BoardExec/Secondary/CrpBEPlinx/CrpBEPlinxWeapon",
|
|
|
|
"/Lotus/Weapons/Corpus/BoardExec/Primary/CrpBEGlaxion/CrpBEGlaxion"
|
|
|
|
];
|