feat(web): типизированный API-клиент, типы DTO, TanStack Query хуки

This commit is contained in:
2026-07-03 17:14:11 +07:00
parent 41242973e1
commit 6f82036e38
5 changed files with 212 additions and 0 deletions
+46
View File
@@ -0,0 +1,46 @@
import { describe, it, expect, vi, beforeEach } from "vitest"
import { api } from "./client"
import { DEFAULT_PROJECT_ID } from "@/lib/config"
beforeEach(() => { vi.restoreAllMocks() })
function mockFetch(body: unknown, ok = true, status = 200) {
return vi.spyOn(globalThis, "fetch").mockResolvedValue({
ok, status,
json: async () => body,
text: async () => JSON.stringify(body),
} as Response)
}
describe("api client", () => {
it("lists accounts at project-scoped path", async () => {
const spy = mockFetch([{ id: "a1", provider: "selectel", comment: "" }])
const accounts = await api.listAccounts()
expect(accounts).toHaveLength(1)
expect(spy).toHaveBeenCalledWith(
`/api/v1/projects/${DEFAULT_PROJECT_ID}/accounts`,
expect.objectContaining({ method: "GET" }),
)
})
it("sends secret on account creation but path has no secret leakage in response typing", async () => {
const spy = mockFetch({ id: "a2", provider: "selectel", comment: "prod" })
await api.createAccount({ provider: "selectel", secret: "TOKEN", comment: "prod" })
const [, opts] = spy.mock.calls[0]
expect((opts as RequestInit).method).toBe("POST")
expect(String((opts as RequestInit).body)).toContain("TOKEN")
})
it("throws on non-ok response", async () => {
mockFetch({ error: "boom" }, false, 500)
await expect(api.listDomains()).rejects.toThrow()
})
it("applies with prune flag", async () => {
const spy = mockFetch({ updates: [], prunes: [], readOnly: [], inSyncCount: 0 })
await api.applyDomain("d1", { applyUpdates: true, applyPrunes: true })
const [url, opts] = spy.mock.calls[0]
expect(url).toContain("/domains/d1/apply")
expect(String((opts as RequestInit).body)).toContain("applyPrunes")
})
})