// The Lucia project. // Copyright 2026-2026 DSP, inc. All rights reserved. // Authors: // imacat.yang@dsp.im (imacat), 2026/03/05 import { describe, it, expect, beforeEach, vi } from "vitest"; import { setActivePinia, createPinia } from "pinia"; vi.mock("@/module/apiError.js", () => ({ default: vi.fn(), })); const { mockGet, mockPost, mockPut } = vi.hoisted(() => ({ mockGet: vi.fn(), mockPost: vi.fn(), mockPut: vi.fn(), })); vi.mock("@/api/client.js", () => ({ default: { get: mockGet, post: mockPost, put: mockPut }, })); import { useAllMapDataStore } from "@/stores/allMapData"; describe("allMapDataStore", () => { let store; beforeEach(() => { setActivePinia(createPinia()); store = useAllMapDataStore(); vi.clearAllMocks(); }); it("has correct default state", () => { expect(store.logId).toBeNull(); expect(store.allProcessMap).toEqual({}); expect(store.allTrace).toEqual([]); }); describe("getAllMapData", () => { it("fetches log discover data", async () => { store.logId = 1; const mockData = { process_map: { nodes: [] }, bpmn: { nodes: [] }, stats: { cases: 10 }, insights: {}, }; mockGet.mockResolvedValue({ data: mockData }); await store.getAllMapData(); expect(mockGet).toHaveBeenCalledWith("/api/logs/1/discover"); expect(store.allProcessMap).toEqual({ nodes: [] }); expect(store.allStats).toEqual({ cases: 10 }); }); it("fetches temp filter discover data when set", async () => { store.logId = 1; store.tempFilterId = 5; mockGet.mockResolvedValue({ data: { process_map: {}, bpmn: {}, stats: {}, insights: {}, }, }); await store.getAllMapData(); expect(mockGet).toHaveBeenCalledWith("/api/temp-filters/5/discover"); }); it("fetches created filter discover data", async () => { store.logId = 1; store.createFilterId = 3; mockGet.mockResolvedValue({ data: { process_map: {}, bpmn: {}, stats: {}, insights: {}, }, }); await store.getAllMapData(); expect(mockGet).toHaveBeenCalledWith("/api/filters/3/discover"); }); it("does not throw on API failure", async () => { store.logId = 1; mockGet.mockRejectedValue(new Error("fail")); await expect(store.getAllMapData()).resolves.not.toThrow(); }); }); describe("getFilterParams", () => { it("fetches filter params and transforms timeframe", async () => { store.logId = 1; const mockData = { tasks: ["A", "B"], sources: ["A"], sinks: ["B"], timeframe: { x_axis: { min: "2023-01-01T00:00:00Z", max: "2023-12-31T00:00:00Z", }, }, trace: [], attrs: [], }; mockGet.mockResolvedValue({ data: mockData }); await store.getFilterParams(); expect(mockGet).toHaveBeenCalledWith("/api/filters/params?log_id=1"); expect(store.allFilterTask).toEqual(["A", "B"]); // Check that min_base and max_base are stored expect(store.allFilterTimeframe.x_axis.min_base).toBe( "2023-01-01T00:00:00Z", ); }); }); describe("checkHasResult", () => { it("posts rule data and stores result", async () => { store.logId = 1; store.postRuleData = [{ type: "task" }]; mockPost.mockResolvedValue({ data: { result: true }, }); await store.checkHasResult(); expect(mockPost).toHaveBeenCalledWith( "/api/filters/has-result?log_id=1", [{ type: "task" }], ); expect(store.hasResultRule).toBe(true); }); }); describe("addTempFilterId", () => { it("creates temp filter and stores id", async () => { store.logId = 1; store.postRuleData = []; mockPost.mockResolvedValue({ data: { id: 77 } }); await store.addTempFilterId(); expect(store.tempFilterId).toBe(77); }); }); describe("addFilterId", () => { it("creates filter and clears temp id", async () => { store.logId = 1; store.tempFilterId = 77; store.postRuleData = [{ type: "rule" }]; mockPost.mockResolvedValue({ data: { id: 88 } }); await store.addFilterId("myFilter"); expect(mockPost).toHaveBeenCalledWith("/api/filters?log_id=1", { name: "myFilter", rules: [{ type: "rule" }], }); expect(store.createFilterId).toBe(88); expect(store.tempFilterId).toBeNull(); }); }); describe("updateFilter", () => { it("updates filter and clears temp id", async () => { store.createFilterId = 88; store.tempFilterId = 77; store.postRuleData = [{ type: "updated" }]; mockPut.mockResolvedValue({ status: 200 }); await store.updateFilter(); expect(mockPut).toHaveBeenCalledWith("/api/filters/88", [ { type: "updated" }, ]); expect(store.isUpdateFilter).toBe(true); expect(store.tempFilterId).toBeNull(); }); }); describe("getAllTrace", () => { it("does not crash when baseLogId is falsy", async () => { store.logId = 1; store.baseLogId = null; mockGet.mockResolvedValue({ data: [{ id: 1 }] }); await store.getAllTrace(); expect(store.allTrace).toEqual([{ id: 1 }]); // allBaseTrace should not be set when baseLogId is falsy expect(store.allBaseTrace).toEqual([]); }); }); describe("getters", () => { it("traces getter sorts by id", () => { store.allTrace = [ { id: 3, name: "C" }, { id: 1, name: "A" }, { id: 2, name: "B" }, ]; expect(store.traces.map((t) => t.id)).toEqual([1, 2, 3]); }); it("processMap getter returns state", () => { store.allProcessMap = { nodes: [1, 2] }; expect(store.processMap).toEqual({ nodes: [1, 2] }); }); it("BaseInfiniteFirstCases returns allBaseCase when baseInfiniteStart is 0", () => { store.baseInfiniteStart = 0; store.allBaseCase = [{ id: 1 }, { id: 2 }]; expect(store.BaseInfiniteFirstCases).toEqual([{ id: 1 }, { id: 2 }]); }); it("BaseInfiniteFirstCases returns undefined when baseInfiniteStart is not 0", () => { store.baseInfiniteStart = 5; store.allBaseCase = [{ id: 1 }]; expect(store.BaseInfiniteFirstCases).toBeUndefined(); }); it("filterAttrs getter does not corrupt original state on repeated access", () => { const originalDate = "2023-06-15T10:30:00Z"; store.allFilterAttrs = [ { type: "date", min: originalDate, max: "2023-12-31T23:59:00Z" }, ]; // Access the getter once store.filterAttrs; // The original state should still contain the ISO date, not a formatted string expect(store.allFilterAttrs[0].min).toBe(originalDate); }); it("filterAttrs getter returns consistent float values on repeated access", () => { store.allFilterAttrs = [{ type: "float", min: 1.239, max: 5.671 }]; const first = store.filterAttrs; const second = store.filterAttrs; // min rounds down (ROUND_DOWN=1), max rounds up (ROUND_UP=0) // Both accesses should produce the same result expect(first[0].min).toBe(second[0].min); expect(first[0].max).toBe(second[0].max); }); }); });