generated from nhcarrigan/template
666a5b2d6d
## What changed and why ### Runestone formula (`prestige.ts`) - Swapped `sqrt` for `cbrt` — much stronger diminishing returns for large gold values - Added base cap of **200** (→ ~1,125 max with all upgrades at 5.625× multiplier) - Prevents extended AFK sessions from producing runestone windfalls that allow immediate upgrade purchasing and rapid prestige chaining ### Prestige threshold formula (`prestige.ts`) - Old: `1,000,000 × 5^n` — exponential, grows impossibly fast, prestige 10+ takes years - New: `1,000,000 × (n+1)²` — polynomial, peaks at ~1 day/run around P8–10, then gets *easier* as the production multiplier overtakes it - Removed `thresholdScaleFactor` constant (no longer needed) ### Production multiplier (`prestige.ts`) - Old: `1.15^n` - New: `1.25^n` — compounds faster, ensures the polynomial threshold eventually gets easy in the late game ### Boss prestige requirements (`bosses.ts`) - Rescaled proportionally from 0–88 range to 0–20 range - The Absolute One now requires prestige **20** (was 88), making transcendence reachable in a few weeks of idle play ### Echo formula (`transcendence.ts`) - Constant changed from 853 → **224** - At the target prestige of 20: `floor(224 / sqrt(20)) = 50 echoes` per transcendence (no meta upgrades) - With all echo_meta upgrades (3.75× total): up to **187 echoes** per transcendence ### Transcendence upgrade costs (`transcendenceUpgrades.ts`) - Old total: **866 echoes** → New total: **400 echoes** (roughly halved across all categories) - Apotheosis still requires **all 15 upgrades** purchased ### Balance fixes (closes #141, #142, #143, #144, #145) - Equipment: `philosophers_stone` click multiplier 2.25→2.5, `crystal_shard` 1.55→1.65 (#144) - Recipes: added `primal_omega_lens` cross-zone click_power recipe at 1.38× (#142) - Adventurers: `celestial_guard` base cost adjusted to smooth tier 14→15→16 cost curve (#145) ### Quest reward rebalancing (closes #136, #137) - Shadow Marshes: buffed `shadow_mere`, `witch_coven`, `plague_ruins` rewards to match combat requirements (#136) - Astral Void: added gold to `void_rift`, increased rewards across all Astral Void quests (#137) ### Boss reward additions (closes #138, #139, #140) - Assigned 9 unassigned adventurer-specific upgrades to Crystalline Spire through Eternal Throne bosses that had empty `upgradeRewards` arrays (#140) ### Combat power documentation (closes #153) - Expanded JSDoc on `computePartyCombatPower` to clarify companion `bossDamage` multiplier behaviour ### Effective adventurer stats (closes #154) - Added `computeEffectiveAdventurerStats` to `tick.ts` and updated `AdventurerCard` to display effective post-multiplier stats ### Adventurer upgrade timing (closes #158) - Audited every adventurer-specific upgrade reward — upgrades now land within the same progression window where that adventurer tier is still a meaningful contributor ### Sync and save fixes (closes #147, #148, #151) - Fixed sync new content count to report only genuinely changed items (#147) - Fixed signature mismatch after first auto-boss completion (#148) - Added auto-buy cap (100) on non-max-tier adventurers (#151) ### Auto-adventurer persistence (closes #156) - Auto-buy preference now preserved across prestige resets ### Broken CDN image (closes #159) - Uploaded missing `auto_adventurer.jpg` to CDN ### Codex unlock hints (closes #146) - Locked codex entries now display a hint generated from `sourceType` and `sourceId` ### Exploration bug fixes (closes #160, #161) - Fixed auto-save race condition discarding exploration materials collected mid-tick (#160) - Fixed exploration areas failing to unlock when zone was unlocked via boss kill or quest completion (#161) ### Concurrent prestige fix (closes #162) - Added optimistic locking via `updatedAt` — concurrent prestige requests return 409 ### Prestige UX (closes #163) - Added `reloadSilent` to game context — no loading screen flash after prestige ### Balance adjustments (closes #164, #165, #166, #167) - Reduced `shadow_mere` CP requirement 5,000,000 → 2,000,000 (#164) - Buffed crystal drops from Shadow Marshes bosses and quests (#165) - Increased runestone yield from 10 → 15 per prestige level (#166) - Daily challenge set always includes a clicks challenge (#167) ### Progression QoL (closes #168, #169) - Added `computeProjectedRunestones()` and persistent `+N On Prestige` resource bar row (#168) - Added `enablePrestigeAnnouncements` setting per player (#169) --- ## Comprehensive balance audit (closes #187, #191, #192, #193, #194, #195, #196, #197, #198) ### Crystal economy fixes - Zeroed crystal rewards for all Zone 7+ boss drops (Celestial Reaches onwards) — crystals are an early/mid-game currency and should not flow freely into the endgame (#187) - Zeroed crystal rewards for all Zone 9+ quest rewards (Infernal Court onwards) — same rationale (#191) ### Achievement additions and fixes - Added quest milestone achievements at 75 quests (10,000 crystals) and 100 quests (15,000 crystals) - Added boss milestone achievement at 50 bosses (15,000 crystals) - Added prestige milestone achievements at P50, P100, P150, P200 — rewarding **runestones** rather than crystals to match the late-game economy - Added gold milestone achievements through 1e90 gold earned - Fixed `quest_eternal` condition from 122 → **112** (actual quest count) — was permanently impossible (#197) - Fixed `fully_equipped` condition from 65 → **78** (actual equipment count after new items) (#197) - Fixed `devourer_slayer` description to remove incorrect zone reference ### Upgrade balance - Fixed Essence Guild multiplier 1.5× → **2×** — was identical to the cheaper Merchant Alliance for 5× the cost (#194) - Raised Void Ascendancy crystal cost 10M → **50M** — was trivially cheap compared to the parallel Celestial Mandate upgrade (100B essence + 50T gold) (#195) - Fixed Sunken Temple quest rewards (gold 2M → 60M, essence 1,500 → 25,000, crystals 75 → 400) — was rewarding less than its easier prerequisite Witch Coven (#193) ### Equipment balance - Buffed Eternal Prism stats to click 5×, combat **3×**, gold **2.5×** — was only marginally better than the free Eternity Stone boss drop for 100M crystals (#196) ### Missing content - Created **13 missing equipment items** for Zones 15–18 (primordial_chaos through the_absolute) that were referenced by late-game boss `equipmentRewards` arrays but never existed in `equipment.ts` (#198): - `chaos_mantle`, `titan_core` (Primordial Chaos) - `expanse_blade`, `void_armour_mk2` (Infinite Expanse) - `cosmos_blade`, `reality_plate` (Reality Forge) - `maelstrom_edge`, `cosmic_plate` (Cosmic Maelstrom) - `primeval_blade`, `ancient_aegis` (Primeval Sanctum) - `absolute_blade`, `eternity_plate`, `omniversal_core` (The Absolute) - Stats scale from combat 14× / gold 9× (Zone 15) up to combat 28× / gold 20× for the final boss drops ### Type system - Extended `AchievementReward` type to support `runestones` field - Updated tick engine achievement processing to award both crystals and runestones --- ## Target progression timeline (optimal play, ~16h/day idle) - First cycle to P20: ~375h (~3.3 weeks) - Each subsequent cycle gets faster as echo upgrades boost income/combat/threshold - Expected **~5 transcendences** before apotheosis at 50–187 echoes/transcendence - **~6 months** to apotheosis for a dedicated player ## Test plan - [ ] Lint, build, and test pipeline passes (100% coverage maintained) - [ ] Prestige threshold at P0 is still 1,000,000 gold - [ ] Prestige runs feel ~1 day long around P8–10 and get easier after - [ ] The Absolute One is locked until prestige 20 - [ ] Transcendence at P20 awards 50 echoes (no meta upgrades) - [ ] All 15 transcendence upgrades cost 400 echoes total - [ ] Bosses in Zones 7+ drop 0 crystals; Zones 1–6 retain crystal drops - [ ] Quests in Zones 9+ reward 0 crystals; Zones 1–8 retain crystal rewards - [ ] Sunken Temple rewards more gold/essence/crystals than Witch Coven - [ ] Essence Guild gives 2× income (stronger than Merchant Alliance 1.5×) - [ ] Void Ascendancy costs 50M crystals - [ ] Eternal Prism stats are click 5×, combat 3×, gold 2.5× - [ ] Late-game bosses (primordial_titan through the_absolute_one) drop equipment on kill - [ ] `quest_eternal` achievement requires 112 quests - [ ] `fully_equipped` achievement requires 78 equipment pieces - [ ] P50/P100/P150/P200 prestige achievements reward runestones - [ ] Adventurer cards show effective post-multiplier stats - [ ] Exploration areas unlock correctly when their zone is unlocked - [ ] Concurrent prestige requests return 409 - [ ] No loading screen flash after prestige - [ ] Daily challenge set always includes a clicks challenge - [ ] Resource bar shows `+N On Prestige` runestone preview ✨ This PR was crafted with help from Hikari~ 🌸 Reviewed-on: #135 Co-authored-by: Hikari <hikari@nhcarrigan.com> Co-committed-by: Hikari <hikari@nhcarrigan.com>
294 lines
12 KiB
TypeScript
294 lines
12 KiB
TypeScript
/**
|
|
* @file Profile routes handling player profile retrieval and updates.
|
|
* @copyright nhcarrigan
|
|
* @license Naomi's Public License
|
|
* @author Naomi Carrigan
|
|
*/
|
|
/* eslint-disable max-lines-per-function -- Route handlers require many steps */
|
|
/* eslint-disable max-statements -- Route handlers require many steps */
|
|
/* eslint-disable complexity -- Route handlers have inherent complexity */
|
|
/* eslint-disable stylistic/key-spacing -- ProfileSettings keys exceed max-len when aligned */
|
|
/* eslint-disable stylistic/max-len -- ProfileSettings key names exceed line length limit */
|
|
/* eslint-disable @typescript-eslint/no-unnecessary-condition -- Defensive checks for runtime nullable fields */
|
|
import {
|
|
DEFAULT_PROFILE_SETTINGS,
|
|
type GameState,
|
|
type ProfileSettings,
|
|
type UpdateProfileRequest,
|
|
} from "@elysium/types";
|
|
import { Hono } from "hono";
|
|
import { gameTitles } from "../data/titles.js";
|
|
import { prisma } from "../db/client.js";
|
|
import { authMiddleware } from "../middleware/auth.js";
|
|
import { logger } from "../services/logger.js";
|
|
import { parseUnlockedTitles } from "../services/titles.js";
|
|
import type { HonoEnvironment } from "../types/hono.js";
|
|
|
|
const profileRouter = new Hono<HonoEnvironment>();
|
|
|
|
const validNumberFormats = new Set([ "suffix", "scientific", "engineering" ]);
|
|
|
|
/**
|
|
* Parses a raw profile settings blob from the database into a typed ProfileSettings object.
|
|
* @param raw - The raw value from the database.
|
|
* @returns A valid ProfileSettings object with defaults for missing fields.
|
|
*/
|
|
const parseProfileSettings = (raw: unknown): ProfileSettings => {
|
|
if (raw === null || typeof raw !== "object" || Array.isArray(raw)) {
|
|
return { ...DEFAULT_PROFILE_SETTINGS };
|
|
}
|
|
/* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- Runtime shape check */
|
|
const rawObject = raw as Record<string, unknown>;
|
|
/* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- Runtime shape check */
|
|
const parsedNumberFormat = rawObject.numberFormat as string;
|
|
const numberFormat = validNumberFormats.has(parsedNumberFormat)
|
|
/* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- Runtime shape check */
|
|
? (parsedNumberFormat as ProfileSettings["numberFormat"])
|
|
: "suffix";
|
|
return {
|
|
enableNotifications: rawObject.enableNotifications === true,
|
|
enablePrestigeAnnouncements: rawObject.enablePrestigeAnnouncements !== false,
|
|
enableSounds: rawObject.enableSounds === true,
|
|
numberFormat: numberFormat,
|
|
showAchievementsUnlocked: rawObject.showAchievementsUnlocked !== false,
|
|
showAdventurersRecruited: rawObject.showAdventurersRecruited !== false,
|
|
showApotheosis: rawObject.showApotheosis !== false,
|
|
showBossesDefeated: rawObject.showBossesDefeated !== false,
|
|
showCurrentClicks: rawObject.showCurrentClicks !== false,
|
|
showCurrentGold: rawObject.showCurrentGold !== false,
|
|
showGuildFounded: rawObject.showGuildFounded !== false,
|
|
showLifetimeAchievementsUnlocked: rawObject.showLifetimeAchievementsUnlocked !== false,
|
|
showLifetimeAdventurersRecruited: rawObject.showLifetimeAdventurersRecruited !== false,
|
|
showLifetimeBossesDefeated: rawObject.showLifetimeBossesDefeated !== false,
|
|
showLifetimeQuestsCompleted: rawObject.showLifetimeQuestsCompleted !== false,
|
|
showOnLeaderboards: rawObject.showOnLeaderboards !== false,
|
|
showPrestige: rawObject.showPrestige !== false,
|
|
showQuestsCompleted: rawObject.showQuestsCompleted !== false,
|
|
showTotalClicks: rawObject.showTotalClicks !== false,
|
|
showTotalGold: rawObject.showTotalGold !== false,
|
|
showTranscendence: rawObject.showTranscendence !== false,
|
|
};
|
|
};
|
|
|
|
/**
|
|
* Resolves a title ID to its display name.
|
|
* @param id - The title ID to resolve.
|
|
* @returns An object with id and name fields.
|
|
*/
|
|
const resolveTitle = (id: string): { id: string; name: string } => {
|
|
const title = gameTitles.find((gameTitle) => {
|
|
return gameTitle.id === id;
|
|
});
|
|
return { id: id, name: title?.name ?? id };
|
|
};
|
|
|
|
profileRouter.get("/:discordId", async(context) => {
|
|
try {
|
|
const { discordId } = context.req.param();
|
|
|
|
const [ player, gameStateRecord ] = await Promise.all([
|
|
prisma.player.findUnique({ where: { discordId } }),
|
|
prisma.gameState.findUnique({ where: { discordId } }),
|
|
]);
|
|
|
|
if (!player) {
|
|
return context.json({ error: "Player not found" }, 404);
|
|
}
|
|
|
|
/* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- Prisma returns JsonValue */
|
|
const state = gameStateRecord?.state as unknown as GameState | undefined;
|
|
const prestigeCount = state?.prestige.count ?? 0;
|
|
const transcendenceCount = state?.transcendence?.count ?? 0;
|
|
const apotheosisCount = state?.apotheosis?.count ?? 0;
|
|
const profileSettings = parseProfileSettings(player.profileSettings);
|
|
|
|
const bossesDefeated
|
|
= state?.bosses.filter((boss) => {
|
|
return boss.status === "defeated";
|
|
}).length ?? 0;
|
|
const questsCompleted
|
|
= state?.quests.filter((quest) => {
|
|
return quest.status === "completed";
|
|
}).length ?? 0;
|
|
|
|
let adventurersRecruited = 0;
|
|
if (state) {
|
|
// eslint-disable-next-line capitalized-comments -- v8 ignore
|
|
/* v8 ignore next 3 -- @preserve */
|
|
for (const adventurer of state.adventurers) {
|
|
adventurersRecruited = adventurersRecruited + adventurer.count;
|
|
}
|
|
}
|
|
|
|
// eslint-disable-next-line capitalized-comments -- v8 ignore
|
|
/* v8 ignore next 3 -- @preserve */
|
|
const achievementsUnlocked = (state?.achievements ?? []).filter((achievement) => {
|
|
return achievement.unlockedAt !== null;
|
|
}).length;
|
|
|
|
const unlockedTitleIds = parseUnlockedTitles(player.unlockedTitles);
|
|
const unlockedTitles = unlockedTitleIds.map((id) => {
|
|
return resolveTitle(id);
|
|
});
|
|
|
|
// eslint-disable-next-line capitalized-comments -- v8 ignore
|
|
/* v8 ignore next 12 -- @preserve */
|
|
const equippedItems = (state?.equipment ?? []).
|
|
filter((item) => {
|
|
return item.owned && item.equipped;
|
|
}).
|
|
map((item) => {
|
|
return {
|
|
bonus: item.bonus,
|
|
name: item.name,
|
|
rarity: item.rarity,
|
|
type: item.type,
|
|
};
|
|
});
|
|
|
|
const completedChapters = state?.story?.completedChapters ?? [];
|
|
|
|
return context.json({
|
|
achievementsUnlocked: achievementsUnlocked,
|
|
activeTitle: player.activeTitle,
|
|
adventurersRecruited: adventurersRecruited,
|
|
apotheosisCount: apotheosisCount,
|
|
avatar: player.avatar,
|
|
bio: player.bio ?? "",
|
|
bossesDefeated: bossesDefeated,
|
|
characterClass: player.characterClass,
|
|
characterName: player.characterName,
|
|
characterRace: player.characterRace ?? "",
|
|
completedChapters: completedChapters,
|
|
createdAt: player.createdAt,
|
|
currentRunClicks: state?.player.totalClicks ?? 0,
|
|
currentRunGold: state?.player.totalGoldEarned ?? 0,
|
|
equippedItems: equippedItems,
|
|
guildDescription: player.guildDescription,
|
|
guildName: player.guildName,
|
|
lifetimeAchievementsUnlocked: player.lifetimeAchievementsUnlocked,
|
|
lifetimeAdventurersRecruited: player.lifetimeAdventurersRecruited,
|
|
lifetimeBossesDefeated: player.lifetimeBossesDefeated,
|
|
lifetimeQuestsCompleted: player.lifetimeQuestsCompleted,
|
|
prestigeCount: prestigeCount,
|
|
profileSettings: profileSettings,
|
|
pronouns: player.pronouns ?? "",
|
|
questsCompleted: questsCompleted,
|
|
totalClicks: player.lifetimeClicks,
|
|
totalGoldEarned: player.lifetimeGoldEarned,
|
|
transcendenceCount: transcendenceCount,
|
|
unlockedTitles: unlockedTitles,
|
|
username: player.username,
|
|
});
|
|
} catch (error) {
|
|
void logger.error(
|
|
"profile_get",
|
|
error instanceof Error
|
|
? error
|
|
: new Error(String(error)),
|
|
);
|
|
return context.json({ error: "Internal server error" }, 500);
|
|
}
|
|
});
|
|
|
|
profileRouter.put("/", authMiddleware, async(context) => {
|
|
try {
|
|
const discordId = context.get("discordId");
|
|
const body = await context.req.json<UpdateProfileRequest>();
|
|
|
|
// eslint-disable-next-line @typescript-eslint/strict-boolean-expressions -- Runtime body validation
|
|
if (!body.characterName) {
|
|
return context.json({ error: "Character name cannot be empty" }, 400);
|
|
}
|
|
|
|
const characterName = body.characterName.trim().slice(0, 32);
|
|
|
|
if (characterName === "") {
|
|
return context.json({ error: "Character name cannot be empty" }, 400);
|
|
}
|
|
|
|
const pronouns = (body.pronouns ?? "").trim().slice(0, 20);
|
|
const characterRace = (body.characterRace ?? "").trim().slice(0, 32);
|
|
const characterClass = (body.characterClass ?? "").trim().slice(0, 32);
|
|
const bio = (body.bio ?? "").trim().slice(0, 200);
|
|
const guildName = (body.guildName ?? "").trim().slice(0, 64);
|
|
const guildDescription = (body.guildDescription ?? "").trim().slice(0, 500);
|
|
// eslint-disable-next-line capitalized-comments -- v8 ignore
|
|
/* v8 ignore next 2 -- @preserve */
|
|
/* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- Runtime shape check */
|
|
const parsedNumberFormat = (body.profileSettings.numberFormat ?? "") as string;
|
|
const numberFormat = validNumberFormats.has(parsedNumberFormat)
|
|
/* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- Runtime shape check */
|
|
? (parsedNumberFormat as ProfileSettings["numberFormat"])
|
|
: "suffix";
|
|
const profileSettings: ProfileSettings = {
|
|
enableNotifications: body.profileSettings.enableNotifications ?? false,
|
|
enablePrestigeAnnouncements: body.profileSettings.enablePrestigeAnnouncements ?? true,
|
|
enableSounds: body.profileSettings.enableSounds ?? false,
|
|
numberFormat: numberFormat,
|
|
showAchievementsUnlocked: body.profileSettings.showAchievementsUnlocked ?? true,
|
|
showAdventurersRecruited: body.profileSettings.showAdventurersRecruited ?? true,
|
|
showApotheosis: body.profileSettings.showApotheosis ?? true,
|
|
showBossesDefeated: body.profileSettings.showBossesDefeated ?? true,
|
|
showCurrentClicks: body.profileSettings.showCurrentClicks ?? true,
|
|
showCurrentGold: body.profileSettings.showCurrentGold ?? true,
|
|
showGuildFounded: body.profileSettings.showGuildFounded ?? true,
|
|
showLifetimeAchievementsUnlocked: body.profileSettings.showLifetimeAchievementsUnlocked ?? true,
|
|
showLifetimeAdventurersRecruited: body.profileSettings.showLifetimeAdventurersRecruited ?? true,
|
|
showLifetimeBossesDefeated: body.profileSettings.showLifetimeBossesDefeated ?? true,
|
|
showLifetimeQuestsCompleted: body.profileSettings.showLifetimeQuestsCompleted ?? true,
|
|
showOnLeaderboards: body.profileSettings.showOnLeaderboards ?? true,
|
|
showPrestige: body.profileSettings.showPrestige ?? true,
|
|
showQuestsCompleted: body.profileSettings.showQuestsCompleted ?? true,
|
|
showTotalClicks: body.profileSettings.showTotalClicks ?? true,
|
|
showTotalGold: body.profileSettings.showTotalGold ?? true,
|
|
showTranscendence: body.profileSettings.showTranscendence ?? true,
|
|
};
|
|
|
|
const activeTitle
|
|
= typeof body.activeTitle === "string"
|
|
? body.activeTitle.slice(0, 64)
|
|
: undefined;
|
|
|
|
const updated = await prisma.player.update({
|
|
data: {
|
|
bio: bio,
|
|
characterClass: characterClass,
|
|
characterName: characterName,
|
|
characterRace: characterRace,
|
|
guildDescription: guildDescription,
|
|
guildName: guildName,
|
|
/* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- Prisma requires object */
|
|
profileSettings: profileSettings as object,
|
|
pronouns: pronouns,
|
|
...activeTitle === undefined
|
|
? {}
|
|
: { activeTitle },
|
|
},
|
|
where: { discordId },
|
|
});
|
|
|
|
return context.json({
|
|
activeTitle: updated.activeTitle,
|
|
bio: updated.bio,
|
|
characterClass: updated.characterClass,
|
|
characterName: updated.characterName,
|
|
characterRace: updated.characterRace,
|
|
guildDescription: updated.guildDescription,
|
|
guildName: updated.guildName,
|
|
profileSettings: profileSettings,
|
|
pronouns: updated.pronouns,
|
|
});
|
|
} catch (error) {
|
|
void logger.error(
|
|
"profile_update",
|
|
error instanceof Error
|
|
? error
|
|
: new Error(String(error)),
|
|
);
|
|
return context.json({ error: "Internal server error" }, 500);
|
|
}
|
|
});
|
|
|
|
export { profileRouter };
|