2025-01-31 17:24:42 +01:00
|
|
|
import { IKeyChainRequest } from "@/src/controllers/api/giveKeyChainTriggeredItemsController";
|
2025-02-19 14:09:47 -08:00
|
|
|
import { isEmptyObject } from "@/src/helpers/general";
|
2025-01-31 17:24:42 +01:00
|
|
|
import { TInventoryDatabaseDocument } from "@/src/models/inventoryModels/inventoryModel";
|
2025-02-19 14:09:47 -08:00
|
|
|
import { createMessage } from "@/src/services/inboxService";
|
2025-03-30 08:13:24 -07:00
|
|
|
import { addItem, addItems, addKeyChainItems, setupKahlSyndicate } from "@/src/services/inventoryService";
|
2025-03-10 16:22:02 -07:00
|
|
|
import {
|
|
|
|
fromStoreItem,
|
|
|
|
getKeyChainMessage,
|
|
|
|
getLevelKeyRewards,
|
|
|
|
getQuestCompletionItems
|
|
|
|
} from "@/src/services/itemDataService";
|
2025-02-24 06:14:47 -08:00
|
|
|
import {
|
|
|
|
IInventoryDatabase,
|
|
|
|
IQuestKeyClient,
|
|
|
|
IQuestKeyDatabase,
|
|
|
|
IQuestStage
|
|
|
|
} from "@/src/types/inventoryTypes/inventoryTypes";
|
2025-01-24 14:13:21 +01:00
|
|
|
import { logger } from "@/src/utils/logger";
|
|
|
|
import { HydratedDocument } from "mongoose";
|
2025-02-19 14:09:47 -08:00
|
|
|
import { ExportKeys } from "warframe-public-export-plus";
|
2025-02-24 08:50:07 -08:00
|
|
|
import { addFixedLevelRewards } from "./missionInventoryUpdateService";
|
2025-02-24 20:56:34 -08:00
|
|
|
import { IInventoryChanges } from "../types/purchaseTypes";
|
2025-01-24 14:13:21 +01:00
|
|
|
|
2025-01-31 17:24:42 +01:00
|
|
|
export interface IUpdateQuestRequest {
|
|
|
|
QuestKeys: Omit<IQuestKeyDatabase, "CompletionDate">[];
|
|
|
|
PS: string;
|
|
|
|
questCompletion: boolean;
|
|
|
|
PlayerShipEvents: unknown[];
|
|
|
|
crossPlaySetting: string;
|
|
|
|
DoQuestReward: boolean;
|
|
|
|
}
|
|
|
|
|
2025-03-10 16:22:02 -07:00
|
|
|
export const updateQuestKey = async (
|
2025-01-24 14:13:21 +01:00
|
|
|
inventory: HydratedDocument<IInventoryDatabase>,
|
|
|
|
questKeyUpdate: IUpdateQuestRequest["QuestKeys"]
|
2025-03-10 16:22:02 -07:00
|
|
|
): Promise<IInventoryChanges> => {
|
2025-01-24 14:13:21 +01:00
|
|
|
if (questKeyUpdate.length > 1) {
|
|
|
|
logger.error(`more than 1 quest key not supported`);
|
|
|
|
throw new Error("more than 1 quest key not supported");
|
|
|
|
}
|
|
|
|
|
2025-02-18 17:14:42 -08:00
|
|
|
const questKeyIndex = inventory.QuestKeys.findIndex(questKey => questKey.ItemType === questKeyUpdate[0].ItemType);
|
|
|
|
|
2025-01-24 14:13:21 +01:00
|
|
|
if (questKeyIndex === -1) {
|
2025-02-18 17:14:42 -08:00
|
|
|
throw new Error(`quest key ${questKeyUpdate[0].ItemType} not found`);
|
2025-01-24 14:13:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
inventory.QuestKeys[questKeyIndex] = questKeyUpdate[0];
|
|
|
|
|
2025-03-10 16:22:02 -07:00
|
|
|
let inventoryChanges: IInventoryChanges = {};
|
2025-01-24 14:13:21 +01:00
|
|
|
if (questKeyUpdate[0].Completed) {
|
|
|
|
inventory.QuestKeys[questKeyIndex].CompletionDate = new Date();
|
2025-03-10 16:22:02 -07:00
|
|
|
|
|
|
|
logger.debug(`completed quest ${questKeyUpdate[0].ItemType} `);
|
|
|
|
const questKeyName = questKeyUpdate[0].ItemType;
|
|
|
|
const questCompletionItems = getQuestCompletionItems(questKeyName);
|
|
|
|
logger.debug(`quest completion items`, questCompletionItems);
|
|
|
|
|
|
|
|
if (questCompletionItems) {
|
|
|
|
inventoryChanges = await addItems(inventory as TInventoryDatabaseDocument, questCompletionItems);
|
|
|
|
}
|
|
|
|
inventory.ActiveQuest = "";
|
2025-03-29 15:35:43 -07:00
|
|
|
|
|
|
|
if (questKeyUpdate[0].ItemType == "/Lotus/Types/Keys/NewWarQuest/NewWarQuestKeyChain") {
|
2025-03-30 08:13:24 -07:00
|
|
|
setupKahlSyndicate(inventory as TInventoryDatabaseDocument);
|
2025-03-29 15:35:43 -07:00
|
|
|
}
|
2025-01-24 14:13:21 +01:00
|
|
|
}
|
2025-03-10 16:22:02 -07:00
|
|
|
return inventoryChanges;
|
2025-01-24 14:13:21 +01:00
|
|
|
};
|
|
|
|
|
2025-01-31 17:24:42 +01:00
|
|
|
export const updateQuestStage = (
|
|
|
|
inventory: TInventoryDatabaseDocument,
|
|
|
|
{ KeyChain, ChainStage }: IKeyChainRequest,
|
|
|
|
questStageUpdate: IQuestStage
|
|
|
|
): void => {
|
|
|
|
const quest = inventory.QuestKeys.find(quest => quest.ItemType === KeyChain);
|
|
|
|
|
|
|
|
if (!quest) {
|
|
|
|
throw new Error(`Quest ${KeyChain} not found in QuestKeys`);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!quest.Progress) {
|
|
|
|
throw new Error(`Progress should always exist when giving keychain triggered items or messages`);
|
|
|
|
}
|
|
|
|
|
|
|
|
const questStage = quest.Progress[ChainStage];
|
|
|
|
|
2025-02-24 20:56:34 -08:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
|
2025-01-31 17:24:42 +01:00
|
|
|
if (!questStage) {
|
|
|
|
const questStageIndex = quest.Progress.push(questStageUpdate) - 1;
|
|
|
|
if (questStageIndex !== ChainStage) {
|
|
|
|
throw new Error(`Quest stage index mismatch: ${questStageIndex} !== ${ChainStage}`);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Object.assign(questStage, questStageUpdate);
|
|
|
|
};
|
2025-02-18 17:14:42 -08:00
|
|
|
|
2025-03-15 03:24:39 -07:00
|
|
|
export const addQuestKey = (
|
|
|
|
inventory: TInventoryDatabaseDocument,
|
|
|
|
questKey: IQuestKeyDatabase
|
|
|
|
): IQuestKeyClient | undefined => {
|
2025-02-18 17:14:42 -08:00
|
|
|
if (inventory.QuestKeys.some(q => q.ItemType === questKey.ItemType)) {
|
2025-02-24 15:59:57 -08:00
|
|
|
logger.warn(`Quest key ${questKey.ItemType} already exists. It will not be added`);
|
2025-02-18 17:14:42 -08:00
|
|
|
return;
|
|
|
|
}
|
2025-03-03 12:48:11 -08:00
|
|
|
|
|
|
|
if (questKey.ItemType == "/Lotus/Types/Keys/InfestedMicroplanetQuest/InfestedMicroplanetQuestKeyChain") {
|
|
|
|
void createMessage(inventory.accountOwnerId.toString(), [
|
|
|
|
{
|
|
|
|
sndr: "/Lotus/Language/Bosses/Loid",
|
|
|
|
icon: "/Lotus/Interface/Icons/Npcs/Entrati/Loid.png",
|
|
|
|
sub: "/Lotus/Language/InfestedMicroplanet/DeimosIntroQuestInboxTitle",
|
|
|
|
msg: "/Lotus/Language/InfestedMicroplanet/DeimosIntroQuestInboxMessage"
|
|
|
|
}
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
2025-02-24 06:14:47 -08:00
|
|
|
const index = inventory.QuestKeys.push(questKey);
|
|
|
|
|
|
|
|
return inventory.QuestKeys[index - 1].toJSON<IQuestKeyClient>();
|
2025-02-18 17:14:42 -08:00
|
|
|
};
|
2025-02-19 14:09:47 -08:00
|
|
|
|
2025-03-15 03:24:39 -07:00
|
|
|
export const completeQuest = async (inventory: TInventoryDatabaseDocument, questKey: string): Promise<void> => {
|
2025-02-24 20:56:34 -08:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
|
2025-02-19 14:09:47 -08:00
|
|
|
const chainStages = ExportKeys[questKey]?.chainStages;
|
|
|
|
|
|
|
|
if (!chainStages) {
|
|
|
|
throw new Error(`Quest ${questKey} does not contain chain stages`);
|
|
|
|
}
|
|
|
|
|
|
|
|
const chainStageTotal = ExportKeys[questKey].chainStages?.length ?? 0;
|
|
|
|
|
|
|
|
const existingQuestKey = inventory.QuestKeys.find(qk => qk.ItemType === questKey);
|
|
|
|
|
|
|
|
if (existingQuestKey?.Completed) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const Progress = Array(chainStageTotal).fill({
|
|
|
|
c: 0,
|
|
|
|
i: false,
|
|
|
|
m: false,
|
|
|
|
b: []
|
|
|
|
} satisfies IQuestStage);
|
|
|
|
|
|
|
|
const completedQuestKey: IQuestKeyDatabase = {
|
|
|
|
ItemType: questKey,
|
|
|
|
Completed: true,
|
|
|
|
unlock: true,
|
|
|
|
Progress: Progress,
|
|
|
|
CompletionDate: new Date()
|
|
|
|
};
|
|
|
|
|
|
|
|
//overwrite current quest progress, might lead to multiple quest item rewards
|
|
|
|
if (existingQuestKey) {
|
|
|
|
existingQuestKey.overwrite(completedQuestKey);
|
|
|
|
//Object.assign(existingQuestKey, completedQuestKey);
|
|
|
|
} else {
|
|
|
|
addQuestKey(inventory, completedQuestKey);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (let i = 0; i < chainStageTotal; i++) {
|
|
|
|
if (chainStages[i].itemsToGiveWhenTriggered.length > 0) {
|
|
|
|
await giveKeyChainItem(inventory, { KeyChain: questKey, ChainStage: i });
|
|
|
|
}
|
|
|
|
|
|
|
|
if (chainStages[i].messageToSendWhenTriggered) {
|
|
|
|
await giveKeyChainMessage(inventory, inventory.accountOwnerId.toString(), {
|
|
|
|
KeyChain: questKey,
|
|
|
|
ChainStage: i
|
|
|
|
});
|
|
|
|
}
|
2025-02-23 12:22:54 -08:00
|
|
|
|
|
|
|
const missionName = chainStages[i].key;
|
|
|
|
if (missionName) {
|
|
|
|
const fixedLevelRewards = getLevelKeyRewards(missionName);
|
2025-02-24 08:50:07 -08:00
|
|
|
//logger.debug(`fixedLevelRewards`, fixedLevelRewards);
|
|
|
|
if (fixedLevelRewards.levelKeyRewards) {
|
|
|
|
const missionRewards: { StoreItem: string; ItemCount: number }[] = [];
|
|
|
|
addFixedLevelRewards(fixedLevelRewards.levelKeyRewards, inventory, missionRewards);
|
|
|
|
|
|
|
|
for (const reward of missionRewards) {
|
2025-03-09 07:42:55 -07:00
|
|
|
await addItem(inventory, fromStoreItem(reward.StoreItem), reward.ItemCount);
|
2025-02-23 12:22:54 -08:00
|
|
|
}
|
2025-02-24 08:50:07 -08:00
|
|
|
} else if (fixedLevelRewards.levelKeyRewards2) {
|
|
|
|
for (const reward of fixedLevelRewards.levelKeyRewards2) {
|
|
|
|
if (reward.rewardType == "RT_CREDITS") {
|
|
|
|
inventory.RegularCredits += reward.amount;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (reward.rewardType == "RT_RESOURCE") {
|
2025-03-09 07:42:55 -07:00
|
|
|
await addItem(inventory, fromStoreItem(reward.itemType), reward.amount);
|
2025-02-24 08:50:07 -08:00
|
|
|
} else {
|
2025-03-09 07:42:55 -07:00
|
|
|
await addItem(inventory, fromStoreItem(reward.itemType));
|
2025-02-24 08:50:07 -08:00
|
|
|
}
|
2025-02-23 12:22:54 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2025-02-19 14:09:47 -08:00
|
|
|
}
|
2025-03-30 08:13:24 -07:00
|
|
|
|
|
|
|
const questCompletionItems = getQuestCompletionItems(questKey);
|
|
|
|
logger.debug(`quest completion items`, questCompletionItems);
|
|
|
|
if (questCompletionItems) {
|
|
|
|
await addItems(inventory, questCompletionItems);
|
|
|
|
}
|
|
|
|
|
2025-02-22 11:10:52 -08:00
|
|
|
inventory.ActiveQuest = "";
|
2025-03-30 08:13:24 -07:00
|
|
|
|
|
|
|
if (questKey == "/Lotus/Types/Keys/NewWarQuest/NewWarQuestKeyChain") {
|
|
|
|
setupKahlSyndicate(inventory);
|
|
|
|
}
|
2025-02-19 14:09:47 -08:00
|
|
|
};
|
|
|
|
|
2025-02-24 20:56:34 -08:00
|
|
|
export const giveKeyChainItem = async (
|
|
|
|
inventory: TInventoryDatabaseDocument,
|
|
|
|
keyChainInfo: IKeyChainRequest
|
|
|
|
): Promise<IInventoryChanges> => {
|
2025-02-19 14:09:47 -08:00
|
|
|
const inventoryChanges = await addKeyChainItems(inventory, keyChainInfo);
|
|
|
|
|
|
|
|
if (isEmptyObject(inventoryChanges)) {
|
|
|
|
throw new Error("inventory changes was empty after getting keychain items: should not happen");
|
|
|
|
}
|
|
|
|
// items were added: update quest stage's i (item was given)
|
|
|
|
updateQuestStage(inventory, keyChainInfo, { i: true });
|
|
|
|
|
2025-02-21 15:46:09 +01:00
|
|
|
return inventoryChanges;
|
|
|
|
|
2025-02-19 14:09:47 -08:00
|
|
|
//TODO: Check whether Wishlist is used to track items which should exist uniquely in the inventory
|
|
|
|
/*
|
|
|
|
some items are added or removed (not sure) to the wishlist, in that case a
|
|
|
|
WishlistChanges: ["/Lotus/Types/Items/ShipFeatureItems/ArsenalFeatureItem"],
|
|
|
|
is added to the response, need to determine for which items this is the case and what purpose this has.
|
|
|
|
*/
|
|
|
|
//{"KeyChain":"/Lotus/Types/Keys/VorsPrize/VorsPrizeQuestKeyChain","ChainStage":0}
|
|
|
|
//{"WishlistChanges":["/Lotus/Types/Items/ShipFeatureItems/ArsenalFeatureItem"],"MiscItems":[{"ItemType":"/Lotus/Types/Items/ShipFeatureItems/ArsenalFeatureItem","ItemCount":1}]}
|
|
|
|
};
|
|
|
|
|
|
|
|
export const giveKeyChainMessage = async (
|
|
|
|
inventory: TInventoryDatabaseDocument,
|
|
|
|
accountId: string,
|
|
|
|
keyChainInfo: IKeyChainRequest
|
2025-02-24 20:56:34 -08:00
|
|
|
): Promise<void> => {
|
2025-02-19 14:09:47 -08:00
|
|
|
const keyChainMessage = getKeyChainMessage(keyChainInfo);
|
|
|
|
|
2025-03-02 04:18:59 -08:00
|
|
|
await createMessage(accountId, [keyChainMessage]);
|
2025-02-19 14:09:47 -08:00
|
|
|
|
|
|
|
updateQuestStage(inventory, keyChainInfo, { m: true });
|
|
|
|
};
|