generated from nhcarrigan/template
chore: fix lint, ensure full CI pipeline passes, add verify checklist
- Fix strict-boolean-expressions in 7 route files (runtime body validation) - Fix no-unnecessary-condition in profile.ts and offlineProgress.ts (defensive null checks) - Extend v8 ignore next-N counts in game.ts to reach 100% coverage - Add CI requirements to CLAUDE.md (lint + build + test must pass before commit) - Add manual verification checklist (verify.md) - Remove progress.md
This commit is contained in:
@@ -5,10 +5,10 @@ import {
|
||||
buildPostApotheosisState,
|
||||
isEligibleForApotheosis,
|
||||
} from "../../src/services/apotheosis.js";
|
||||
import { DEFAULT_TRANSCENDENCE_UPGRADES } from "../../src/data/transcendenceUpgrades.js";
|
||||
import { defaultTranscendenceUpgrades } from "../../src/data/transcendenceUpgrades.js";
|
||||
import type { GameState } from "@elysium/types";
|
||||
|
||||
const ALL_UPGRADE_IDS = DEFAULT_TRANSCENDENCE_UPGRADES.map((u) => u.id);
|
||||
const ALL_UPGRADE_IDS = defaultTranscendenceUpgrades.map((u) => u.id);
|
||||
|
||||
const makeMinimalState = (overrides: Partial<GameState> = {}): GameState =>
|
||||
({
|
||||
@@ -74,42 +74,42 @@ describe("isEligibleForApotheosis", () => {
|
||||
describe("buildPostApotheosisState", () => {
|
||||
it("increments apotheosis count from 0", () => {
|
||||
const state = makeMinimalState();
|
||||
const { newApotheosisData } = buildPostApotheosisState(state, "T");
|
||||
expect(newApotheosisData.count).toBe(1);
|
||||
const { updatedApotheosisData } = buildPostApotheosisState(state, "T");
|
||||
expect(updatedApotheosisData.count).toBe(1);
|
||||
});
|
||||
|
||||
it("increments apotheosis count from existing value", () => {
|
||||
const state = makeMinimalState({ apotheosis: { count: 2 } });
|
||||
const { newApotheosisData } = buildPostApotheosisState(state, "T");
|
||||
expect(newApotheosisData.count).toBe(3);
|
||||
const { updatedApotheosisData } = buildPostApotheosisState(state, "T");
|
||||
expect(updatedApotheosisData.count).toBe(3);
|
||||
});
|
||||
|
||||
it("persists codex", () => {
|
||||
const codex = { entries: [{ id: "e1", unlockedAt: 1000, sourceType: "exploration" as const }] };
|
||||
const state = makeMinimalState({ codex });
|
||||
const { newState } = buildPostApotheosisState(state, "T");
|
||||
expect(newState.codex).toEqual(codex);
|
||||
const { updatedState } = buildPostApotheosisState(state, "T");
|
||||
expect(updatedState.codex).toEqual(codex);
|
||||
});
|
||||
|
||||
it("persists story", () => {
|
||||
const story = { unlockedChapterIds: ["ch1"], completedChapters: [] };
|
||||
const state = makeMinimalState({ story });
|
||||
const { newState } = buildPostApotheosisState(state, "T");
|
||||
expect(newState.story).toEqual(story);
|
||||
const { updatedState } = buildPostApotheosisState(state, "T");
|
||||
expect(updatedState.story).toEqual(story);
|
||||
});
|
||||
|
||||
it("wipes prestige data", () => {
|
||||
const state = makeMinimalState({
|
||||
prestige: { count: 10, runestones: 1000, productionMultiplier: 3, purchasedUpgradeIds: [] },
|
||||
});
|
||||
const { newState } = buildPostApotheosisState(state, "T");
|
||||
expect(newState.prestige.count).toBe(0);
|
||||
expect(newState.prestige.runestones).toBe(0);
|
||||
const { updatedState } = buildPostApotheosisState(state, "T");
|
||||
expect(updatedState.prestige.count).toBe(0);
|
||||
expect(updatedState.prestige.runestones).toBe(0);
|
||||
});
|
||||
|
||||
it("sets apotheosis count on new state", () => {
|
||||
const state = makeMinimalState({ apotheosis: { count: 0 } });
|
||||
const { newState } = buildPostApotheosisState(state, "T");
|
||||
expect(newState.apotheosis?.count).toBe(1);
|
||||
const { updatedState } = buildPostApotheosisState(state, "T");
|
||||
expect(updatedState.apotheosis?.count).toBe(1);
|
||||
});
|
||||
});
|
||||
|
||||
@@ -90,19 +90,19 @@ describe("isEligibleForPrestige", () => {
|
||||
describe("calculateRunestones", () => {
|
||||
it("calculates basic runestones formula", () => {
|
||||
// floor(sqrt(4_000_000 / 1_000_000)) × 10 = floor(2) × 10 = 20
|
||||
const result = calculateRunestones(4_000_000, 0, []);
|
||||
const result = calculateRunestones({ totalGoldEarned: 4_000_000, prestigeCount: 0, purchasedUpgradeIds: [] });
|
||||
expect(result).toBe(20);
|
||||
});
|
||||
|
||||
it("applies echo runestone multiplier", () => {
|
||||
// floor(sqrt(4) × 10) = 20; × 2 = 40
|
||||
const result = calculateRunestones(4_000_000, 0, [], 2);
|
||||
const result = calculateRunestones({ totalGoldEarned: 4_000_000, prestigeCount: 0, purchasedUpgradeIds: [], echoRunestoneMultiplier: 2 });
|
||||
expect(result).toBe(40);
|
||||
});
|
||||
|
||||
it("applies purchased runestone upgrade multiplier", () => {
|
||||
// With "runestones_1" purchased (multiplier 1.25): floor(20 × 1.25) = 25
|
||||
const result = calculateRunestones(4_000_000, 0, ["runestone_gain_1"]);
|
||||
const result = calculateRunestones({ totalGoldEarned: 4_000_000, prestigeCount: 0, purchasedUpgradeIds: ["runestone_gain_1"] });
|
||||
expect(result).toBeGreaterThan(20);
|
||||
});
|
||||
});
|
||||
@@ -176,15 +176,15 @@ describe("computeRunestoneMultipliers", () => {
|
||||
describe("buildPostPrestigeState", () => {
|
||||
it("increments prestige count", () => {
|
||||
const state = makeMinimalState({ player: makePlayer(4_000_000) });
|
||||
const { newPrestigeData } = buildPostPrestigeState(state, "Tester");
|
||||
expect(newPrestigeData.count).toBe(1);
|
||||
const { prestigeData } = buildPostPrestigeState(state, "Tester");
|
||||
expect(prestigeData.count).toBe(1);
|
||||
});
|
||||
|
||||
it("sums runestones earned", () => {
|
||||
const state = makeMinimalState({ player: makePlayer(4_000_000) });
|
||||
const { newPrestigeData, runestonesEarned } = buildPostPrestigeState(state, "Tester");
|
||||
const { prestigeData, runestonesEarned } = buildPostPrestigeState(state, "Tester");
|
||||
expect(runestonesEarned).toBeGreaterThan(0);
|
||||
expect(newPrestigeData.runestones).toBe(runestonesEarned);
|
||||
expect(prestigeData.runestones).toBe(runestonesEarned);
|
||||
});
|
||||
|
||||
it("adds milestone runestones at prestige 5", () => {
|
||||
@@ -199,15 +199,15 @@ describe("buildPostPrestigeState", () => {
|
||||
it("persists codex from current state", () => {
|
||||
const codex = { entries: [{ id: "e1", unlockedAt: 1000, sourceType: "exploration" as const }] };
|
||||
const state = makeMinimalState({ codex });
|
||||
const { newState } = buildPostPrestigeState(state, "Tester");
|
||||
expect(newState.codex).toEqual(codex);
|
||||
const { prestigeState } = buildPostPrestigeState(state, "Tester");
|
||||
expect(prestigeState.codex).toEqual(codex);
|
||||
});
|
||||
|
||||
it("persists story from current state", () => {
|
||||
const story = { unlockedChapterIds: ["ch1"], completedChapters: [] };
|
||||
const state = makeMinimalState({ story });
|
||||
const { newState } = buildPostPrestigeState(state, "Tester");
|
||||
expect(newState.story).toEqual(story);
|
||||
const { prestigeState } = buildPostPrestigeState(state, "Tester");
|
||||
expect(prestigeState.story).toEqual(story);
|
||||
});
|
||||
|
||||
it("persists transcendence from current state", () => {
|
||||
@@ -218,28 +218,28 @@ describe("buildPostPrestigeState", () => {
|
||||
echoPrestigeRunestoneMultiplier: 1, echoMetaMultiplier: 1,
|
||||
};
|
||||
const state = makeMinimalState({ transcendence });
|
||||
const { newState } = buildPostPrestigeState(state, "Tester");
|
||||
expect(newState.transcendence).toEqual(transcendence);
|
||||
const { prestigeState } = buildPostPrestigeState(state, "Tester");
|
||||
expect(prestigeState.transcendence).toEqual(transcendence);
|
||||
});
|
||||
|
||||
it("preserves autoPrestigeEnabled when set", () => {
|
||||
const state = makeMinimalState({
|
||||
prestige: { count: 0, runestones: 0, productionMultiplier: 1, purchasedUpgradeIds: [], autoPrestigeEnabled: true },
|
||||
});
|
||||
const { newPrestigeData } = buildPostPrestigeState(state, "Tester");
|
||||
expect(newPrestigeData.autoPrestigeEnabled).toBe(true);
|
||||
const { prestigeData } = buildPostPrestigeState(state, "Tester");
|
||||
expect(prestigeData.autoPrestigeEnabled).toBe(true);
|
||||
});
|
||||
|
||||
it("omits autoPrestigeEnabled when not set", () => {
|
||||
const state = makeMinimalState();
|
||||
const { newPrestigeData } = buildPostPrestigeState(state, "Tester");
|
||||
expect(newPrestigeData.autoPrestigeEnabled).toBeUndefined();
|
||||
const { prestigeData } = buildPostPrestigeState(state, "Tester");
|
||||
expect(prestigeData.autoPrestigeEnabled).toBeUndefined();
|
||||
});
|
||||
|
||||
it("preserves apotheosis data across prestige", () => {
|
||||
const apotheosis = { count: 2 };
|
||||
const state = makeMinimalState({ apotheosis });
|
||||
const { newState } = buildPostPrestigeState(state, "Tester");
|
||||
expect(newState.apotheosis).toEqual(apotheosis);
|
||||
const { prestigeState } = buildPostPrestigeState(state, "Tester");
|
||||
expect(prestigeState.apotheosis).toEqual(apotheosis);
|
||||
});
|
||||
});
|
||||
|
||||
@@ -62,7 +62,7 @@ describe("checkAndUnlockTitles", () => {
|
||||
|
||||
it("returns empty array when no new titles are earned", () => {
|
||||
const state = makeMinimalState();
|
||||
const result = checkAndUnlockTitles([], state, "", NOW);
|
||||
const result = checkAndUnlockTitles({ currentUnlocked: [], state, guildName: "", createdAt: NOW });
|
||||
expect(result).toEqual([]);
|
||||
});
|
||||
|
||||
@@ -70,19 +70,19 @@ describe("checkAndUnlockTitles", () => {
|
||||
const state = makeMinimalState({
|
||||
player: { discordId: "t", username: "u", discriminator: "0", avatar: null, totalGoldEarned: 0, totalClicks: 10_000, characterName: "T" },
|
||||
});
|
||||
const result = checkAndUnlockTitles(["click_maniac"], state, "", NOW);
|
||||
const result = checkAndUnlockTitles({ currentUnlocked: ["click_maniac"], state, guildName: "", createdAt: NOW });
|
||||
expect(result).not.toContain("click_maniac");
|
||||
});
|
||||
|
||||
it("unlocks guild_founder when guild name is non-empty", () => {
|
||||
const state = makeMinimalState();
|
||||
const result = checkAndUnlockTitles([], state, "My Guild", NOW);
|
||||
const result = checkAndUnlockTitles({ currentUnlocked: [], state, guildName: "My Guild", createdAt: NOW });
|
||||
expect(result).toContain("guild_founder");
|
||||
});
|
||||
|
||||
it("does not unlock guild_founder for whitespace-only guild name", () => {
|
||||
const state = makeMinimalState();
|
||||
const result = checkAndUnlockTitles([], state, " ", NOW);
|
||||
const result = checkAndUnlockTitles({ currentUnlocked: [], state, guildName: " ", createdAt: NOW });
|
||||
expect(result).not.toContain("guild_founder");
|
||||
});
|
||||
|
||||
@@ -90,7 +90,7 @@ describe("checkAndUnlockTitles", () => {
|
||||
const state = makeMinimalState({
|
||||
quests: [{ status: "completed" }] as GameState["quests"],
|
||||
});
|
||||
const result = checkAndUnlockTitles([], state, "", NOW);
|
||||
const result = checkAndUnlockTitles({ currentUnlocked: [], state, guildName: "", createdAt: NOW });
|
||||
expect(result).toContain("the_adventurous");
|
||||
});
|
||||
|
||||
@@ -98,7 +98,7 @@ describe("checkAndUnlockTitles", () => {
|
||||
const state = makeMinimalState({
|
||||
bosses: [{ status: "defeated" }] as GameState["bosses"],
|
||||
});
|
||||
const result = checkAndUnlockTitles([], state, "", NOW);
|
||||
const result = checkAndUnlockTitles({ currentUnlocked: [], state, guildName: "", createdAt: NOW });
|
||||
expect(result).toContain("boss_slayer");
|
||||
});
|
||||
|
||||
@@ -106,21 +106,21 @@ describe("checkAndUnlockTitles", () => {
|
||||
const state = makeMinimalState({
|
||||
prestige: { count: 1, runestones: 0, productionMultiplier: 1, purchasedUpgradeIds: [] },
|
||||
});
|
||||
const result = checkAndUnlockTitles([], state, "", NOW);
|
||||
const result = checkAndUnlockTitles({ currentUnlocked: [], state, guildName: "", createdAt: NOW });
|
||||
expect(result).toContain("the_undying");
|
||||
});
|
||||
|
||||
it("unlocks veteran after 30 days of play", () => {
|
||||
const createdAt = NOW - THIRTY_DAYS_MS;
|
||||
const state = makeMinimalState();
|
||||
const result = checkAndUnlockTitles([], state, "", createdAt);
|
||||
const result = checkAndUnlockTitles({ currentUnlocked: [], state, guildName: "", createdAt });
|
||||
expect(result).toContain("veteran");
|
||||
});
|
||||
|
||||
it("does not unlock veteran before 30 days", () => {
|
||||
const createdAt = NOW - (29 * 86_400_000);
|
||||
const state = makeMinimalState();
|
||||
const result = checkAndUnlockTitles([], state, "", createdAt);
|
||||
const result = checkAndUnlockTitles({ currentUnlocked: [], state, guildName: "", createdAt });
|
||||
expect(result).not.toContain("veteran");
|
||||
});
|
||||
|
||||
@@ -129,7 +129,7 @@ describe("checkAndUnlockTitles", () => {
|
||||
bosses: [{ status: "defeated" }] as GameState["bosses"],
|
||||
quests: [{ status: "completed" }] as GameState["quests"],
|
||||
});
|
||||
const result = checkAndUnlockTitles([], state, "Guild", NOW);
|
||||
const result = checkAndUnlockTitles({ currentUnlocked: [], state, guildName: "Guild", createdAt: NOW });
|
||||
expect(result).toContain("boss_slayer");
|
||||
expect(result).toContain("the_adventurous");
|
||||
expect(result).toContain("guild_founder");
|
||||
@@ -145,7 +145,7 @@ describe("checkAndUnlockTitles", () => {
|
||||
apotheosis: { count: 1 },
|
||||
});
|
||||
// Just verify this runs without error — the counts are read via ?. chains
|
||||
const result = checkAndUnlockTitles([], state, "", NOW);
|
||||
const result = checkAndUnlockTitles({ currentUnlocked: [], state, guildName: "", createdAt: NOW });
|
||||
expect(Array.isArray(result)).toBe(true);
|
||||
});
|
||||
});
|
||||
|
||||
@@ -123,8 +123,8 @@ describe("calculateEchoes", () => {
|
||||
describe("buildPostTranscendenceState", () => {
|
||||
it("increments transcendence count from 0", () => {
|
||||
const state = makeMinimalState();
|
||||
const { newTranscendenceData } = buildPostTranscendenceState(state, "T");
|
||||
expect(newTranscendenceData.count).toBe(1);
|
||||
const { transcendenceData } = buildPostTranscendenceState(state, "T");
|
||||
expect(transcendenceData.count).toBe(1);
|
||||
});
|
||||
|
||||
it("accumulates echoes", () => {
|
||||
@@ -135,37 +135,37 @@ describe("buildPostTranscendenceState", () => {
|
||||
echoPrestigeThresholdMultiplier: 1, echoPrestigeRunestoneMultiplier: 1, echoMetaMultiplier: 1,
|
||||
},
|
||||
});
|
||||
const { newTranscendenceData, echoesEarned } = buildPostTranscendenceState(state, "T");
|
||||
expect(newTranscendenceData.echoes).toBe(100 + echoesEarned);
|
||||
const { transcendenceData, echoesEarned } = buildPostTranscendenceState(state, "T");
|
||||
expect(transcendenceData.echoes).toBe(100 + echoesEarned);
|
||||
});
|
||||
|
||||
it("persists codex from current state", () => {
|
||||
const codex = { entries: [{ id: "e1", unlockedAt: 1000, sourceType: "exploration" as const }] };
|
||||
const state = makeMinimalState({ codex });
|
||||
const { newState } = buildPostTranscendenceState(state, "T");
|
||||
expect(newState.codex).toEqual(codex);
|
||||
const { transcendenceState } = buildPostTranscendenceState(state, "T");
|
||||
expect(transcendenceState.codex).toEqual(codex);
|
||||
});
|
||||
|
||||
it("persists story from current state", () => {
|
||||
const story = { unlockedChapterIds: ["ch1"], completedChapters: [] };
|
||||
const state = makeMinimalState({ story });
|
||||
const { newState } = buildPostTranscendenceState(state, "T");
|
||||
expect(newState.story).toEqual(story);
|
||||
const { transcendenceState } = buildPostTranscendenceState(state, "T");
|
||||
expect(transcendenceState.story).toEqual(story);
|
||||
});
|
||||
|
||||
it("persists apotheosis from current state", () => {
|
||||
const apotheosis = { count: 2 };
|
||||
const state = makeMinimalState({ apotheosis });
|
||||
const { newState } = buildPostTranscendenceState(state, "T");
|
||||
expect(newState.apotheosis).toEqual(apotheosis);
|
||||
const { transcendenceState } = buildPostTranscendenceState(state, "T");
|
||||
expect(transcendenceState.apotheosis).toEqual(apotheosis);
|
||||
});
|
||||
|
||||
it("resets prestige to fresh state", () => {
|
||||
const state = makeMinimalState({
|
||||
prestige: { count: 5, runestones: 500, productionMultiplier: 2, purchasedUpgradeIds: [] },
|
||||
});
|
||||
const { newState } = buildPostTranscendenceState(state, "T");
|
||||
expect(newState.prestige.count).toBe(0);
|
||||
expect(newState.prestige.runestones).toBe(0);
|
||||
const { transcendenceState } = buildPostTranscendenceState(state, "T");
|
||||
expect(transcendenceState.prestige.count).toBe(0);
|
||||
expect(transcendenceState.prestige.runestones).toBe(0);
|
||||
});
|
||||
});
|
||||
|
||||
Reference in New Issue
Block a user