aboutsummaryrefslogtreecommitdiffhomepage
path: root/src/client/components/DeleteCardModal.test.tsx
diff options
context:
space:
mode:
authornsfisis <nsfisis@gmail.com>2026-01-01 23:44:50 +0900
committernsfisis <nsfisis@gmail.com>2026-01-01 23:47:21 +0900
commit2fb6471a685bec1433be3335f377a1a2313e4820 (patch)
tree328ddaeec0c591b06bf005d48b0242345c1336be /src/client/components/DeleteCardModal.test.tsx
parentf30566e1c7126db4c6242ab38d07a9478f79d3db (diff)
downloadkioku-2fb6471a685bec1433be3335f377a1a2313e4820.tar.gz
kioku-2fb6471a685bec1433be3335f377a1a2313e4820.tar.zst
kioku-2fb6471a685bec1433be3335f377a1a2313e4820.zip
refactor(client): migrate API calls to typed RPC client
Replace raw fetch() calls with apiClient.rpc typed client across all modal and page components. This provides better type safety and eliminates manual auth header handling. - Make handleResponse public for component usage - Update all component tests to mock RPC methods instead of fetch - Change POSTGRES_HOST default to kioku-db for Docker compatibility 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
Diffstat (limited to 'src/client/components/DeleteCardModal.test.tsx')
-rw-r--r--src/client/components/DeleteCardModal.test.tsx70
1 files changed, 33 insertions, 37 deletions
diff --git a/src/client/components/DeleteCardModal.test.tsx b/src/client/components/DeleteCardModal.test.tsx
index 4178ee8..6536c6f 100644
--- a/src/client/components/DeleteCardModal.test.tsx
+++ b/src/client/components/DeleteCardModal.test.tsx
@@ -4,11 +4,26 @@
import { cleanup, render, screen, waitFor } from "@testing-library/react";
import userEvent from "@testing-library/user-event";
import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
-import { apiClient } from "../api/client";
+
+const mockDelete = vi.fn();
+const mockHandleResponse = vi.fn();
vi.mock("../api/client", () => ({
apiClient: {
- getAuthHeader: vi.fn(),
+ rpc: {
+ api: {
+ decks: {
+ ":deckId": {
+ cards: {
+ ":cardId": {
+ $delete: (args: unknown) => mockDelete(args),
+ },
+ },
+ },
+ },
+ },
+ },
+ handleResponse: (res: unknown) => mockHandleResponse(res),
},
ApiClientError: class ApiClientError extends Error {
constructor(
@@ -22,13 +37,10 @@ vi.mock("../api/client", () => ({
},
}));
+import { ApiClientError } from "../api/client";
// Import after mock is set up
import { DeleteCardModal } from "./DeleteCardModal";
-// Mock fetch globally
-const mockFetch = vi.fn();
-global.fetch = mockFetch;
-
describe("DeleteCardModal", () => {
const mockCard = {
id: "card-123",
@@ -45,9 +57,8 @@ describe("DeleteCardModal", () => {
beforeEach(() => {
vi.clearAllMocks();
- vi.mocked(apiClient.getAuthHeader).mockReturnValue({
- Authorization: "Bearer access-token",
- });
+ mockDelete.mockResolvedValue({ ok: true });
+ mockHandleResponse.mockResolvedValue({});
});
afterEach(() => {
@@ -142,11 +153,6 @@ describe("DeleteCardModal", () => {
const onClose = vi.fn();
const onCardDeleted = vi.fn();
- mockFetch.mockResolvedValue({
- ok: true,
- json: async () => ({}),
- });
-
render(
<DeleteCardModal
isOpen={true}
@@ -160,15 +166,9 @@ describe("DeleteCardModal", () => {
await user.click(screen.getByRole("button", { name: "Delete" }));
await waitFor(() => {
- expect(mockFetch).toHaveBeenCalledWith(
- "/api/decks/deck-456/cards/card-123",
- {
- method: "DELETE",
- headers: {
- Authorization: "Bearer access-token",
- },
- },
- );
+ expect(mockDelete).toHaveBeenCalledWith({
+ param: { deckId: "deck-456", cardId: "card-123" },
+ });
});
expect(onCardDeleted).toHaveBeenCalledTimes(1);
@@ -178,7 +178,7 @@ describe("DeleteCardModal", () => {
it("shows loading state during deletion", async () => {
const user = userEvent.setup();
- mockFetch.mockImplementation(() => new Promise(() => {})); // Never resolves
+ mockDelete.mockImplementation(() => new Promise(() => {})); // Never resolves
render(<DeleteCardModal {...defaultProps} />);
@@ -198,11 +198,9 @@ describe("DeleteCardModal", () => {
it("displays API error message", async () => {
const user = userEvent.setup();
- mockFetch.mockResolvedValue({
- ok: false,
- status: 404,
- json: async () => ({ error: "Card not found" }),
- });
+ mockHandleResponse.mockRejectedValue(
+ new ApiClientError("Card not found", 404),
+ );
render(<DeleteCardModal {...defaultProps} />);
@@ -216,7 +214,7 @@ describe("DeleteCardModal", () => {
it("displays generic error on unexpected failure", async () => {
const user = userEvent.setup();
- mockFetch.mockRejectedValue(new Error("Network error"));
+ mockDelete.mockRejectedValue(new Error("Network error"));
render(<DeleteCardModal {...defaultProps} />);
@@ -229,10 +227,12 @@ describe("DeleteCardModal", () => {
});
});
- it("displays error when not authenticated", async () => {
+ it("displays error when handleResponse throws", async () => {
const user = userEvent.setup();
- vi.mocked(apiClient.getAuthHeader).mockReturnValue(undefined);
+ mockHandleResponse.mockRejectedValue(
+ new ApiClientError("Not authenticated", 401),
+ );
render(<DeleteCardModal {...defaultProps} />);
@@ -249,11 +249,7 @@ describe("DeleteCardModal", () => {
const user = userEvent.setup();
const onClose = vi.fn();
- mockFetch.mockResolvedValue({
- ok: false,
- status: 404,
- json: async () => ({ error: "Some error" }),
- });
+ mockHandleResponse.mockRejectedValue(new ApiClientError("Some error", 404));
const { rerender } = render(
<DeleteCardModal {...defaultProps} onClose={onClose} />,