Skip to content

Commit

Permalink
[UI v2] exercise: Adds test for variables hook that focuses on react …
Browse files Browse the repository at this point in the history
…query and query caching and invalidation
  • Loading branch information
devinvillarosa committed Dec 1, 2024
1 parent 1aa0186 commit 475a68f
Show file tree
Hide file tree
Showing 2 changed files with 284 additions and 2 deletions.
282 changes: 282 additions & 0 deletions ui-v2/src/hooks/variables.test.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,282 @@
import { http, HttpResponse } from "msw";
import { describe, it, expect } from "vitest";
import { act, renderHook, waitFor } from "@testing-library/react";
import { QueryClient, QueryClientProvider } from "@tanstack/react-query";

import {
buildCountQuery,
buildVariablesQuery,
useCreateVariable,
useDeleteVariable,
useVariables,
useUpdateVariable,
} from "./variables";

import { server } from "../../tests/mocks/node";

describe("variable hooks", () => {
const variableFilter = {
offset: 0,
sort: "CREATED_DESC" as const,
};

const initialSeedVariables = () => [
{
id: "1",
name: "my-variable",
value: 123,
created: "2021-01-01T00:00:00Z",
updated: "2021-01-01T00:00:00Z",
tags: ["tag1"],
},
];

const createQueryWrapper = ({ queryClient = new QueryClient() }) => {
const QueryWrapper = ({ children }: { children: React.ReactNode }) => (
<QueryClientProvider client={queryClient}>{children}</QueryClientProvider>
);
return QueryWrapper;
};

/**
* Data Management:
* - Asserts variable data and count is fetched based on the APIs invoked for the hook
*/
it("useVariables() is able to fetch from initial cache", async () => {
const queryClient = new QueryClient();

// ------------ Mock API requests
const mockFetchVariablesAPIs = () => {
const retVariables = initialSeedVariables();
server.use(
http.post("http://localhost:4200/api/variables/filter", () => {
return HttpResponse.json(retVariables);
}),
http.post("http://localhost:4200/api/variables/count", () => {
return HttpResponse.json(retVariables.length);
}),
);
};
mockFetchVariablesAPIs();

// ------------ Initialize hooks to test
const { result } = renderHook(() => useVariables(variableFilter), {
wrapper: createQueryWrapper({ queryClient }),
});

// ------------ Assert
await waitFor(() => expect(result.current.isLoading).toBe(false));
expect(result.current.isError).toBe(false);
expect(result.current.totalCount).toEqual(1);
});

/**
* Data Management:
* - Asserts delete mutation API is called.
* - Upon delete mutation API being called, cache is invalidated and asserts cache invalidation APIS are called
*/
it("useDeleteVariable() invalidates cache and fetches updated value", async () => {
const VARIABLE_ID_TO_DELETE = "1";
const queryClient = new QueryClient();

// ------------ Mock API requests
const mockDeleteVariableAPIs = () => {
const variableData = initialSeedVariables().filter(
(variable) => variable.id !== VARIABLE_ID_TO_DELETE,
);
server.use(
http.delete("http://localhost:4200/api/variables/:id", () => {
return HttpResponse.json();
}),
http.post("http://localhost:4200/api/variables/filter", () => {
return HttpResponse.json(variableData);
}),
http.post("http://localhost:4200/api/variables/count", () => {
return HttpResponse.json(variableData.length);
}),
);
};
mockDeleteVariableAPIs();

// ------------ Initialize cache
queryClient.setQueryData(
buildVariablesQuery(variableFilter).queryKey,
initialSeedVariables(),
);
queryClient.setQueryData(
buildCountQuery(variableFilter).queryKey,
initialSeedVariables().length,
);

// ------------ Initialize hooks to test
const { result: useVariablesResult } = renderHook(
() => useVariables(variableFilter),
{ wrapper: createQueryWrapper({ queryClient }) },
);

const { result: useDeleteVariableResult } = renderHook(useDeleteVariable, {
wrapper: createQueryWrapper({ queryClient }),
});

// ------------ Invoke mutation
act(() =>
useDeleteVariableResult.current.deleteVariable(VARIABLE_ID_TO_DELETE),
);

// ------------ Assert
await waitFor(() =>
expect(useDeleteVariableResult.current.isSuccess).toBe(true),
);
expect(useVariablesResult.current.variables.length).toEqual(0);
expect(useVariablesResult.current.totalCount).toEqual(0);
});

/**
* Data Management:
* - Asserts create mutation API is called.
* - Upon create mutation API being called, cache is invalidated and asserts cache invalidation APIS are called
*/
it("useCreateVariable() invalidates cache and fetches updated value", async () => {
const queryClient = new QueryClient();
const MOCK_NEW_VARIABLE_ID = "2";
const MOCK_NEW_VARIABLE = {
name: "my-new-variable",
value: 123,
created: "2021-01-01T00:00:00Z",
updated: "2021-01-01T00:00:00Z",
tags: ["tag1"],
};

// ------------ Mock API requests
const mockCreateVariableAPIs = () => {
const variableData = [
...initialSeedVariables(),
{
...MOCK_NEW_VARIABLE,
id: MOCK_NEW_VARIABLE_ID,
},
];
server.use(
http.post("http://localhost:4200/api/variables/", () => {
return HttpResponse.json();
}),
http.post("http://localhost:4200/api/variables/filter", () => {
return HttpResponse.json(variableData);
}),
http.post("http://localhost:4200/api/variables/count", () => {
return HttpResponse.json(variableData.length);
}),
);
};
mockCreateVariableAPIs();

// ------------ Initialize cache
queryClient.setQueryData(
buildVariablesQuery(variableFilter).queryKey,
initialSeedVariables(),
);
queryClient.setQueryData(
buildCountQuery(variableFilter).queryKey,
initialSeedVariables().length,
);

// ------------ Initialize hooks to test
const { result: useVariablesResult } = renderHook(
() => useVariables(variableFilter),
{ wrapper: createQueryWrapper({ queryClient }) },
);
const { result: useCreateVariableResult } = renderHook(useCreateVariable, {
wrapper: createQueryWrapper({ queryClient }),
});

// ------------ Invoke mutation
act(() =>
useCreateVariableResult.current.createVariable(MOCK_NEW_VARIABLE),
);

// ------------ Assert
await waitFor(() =>
expect(useCreateVariableResult.current.isSuccess).toBe(true),
);
expect(useVariablesResult.current.variables.length).toEqual(2);
expect(useVariablesResult.current.totalCount).toEqual(2);
const newVariable = useVariablesResult.current.variables.find(
(variable) => variable.id === MOCK_NEW_VARIABLE_ID,
);
expect(newVariable).toBeTruthy();
});

/**
* Data Management:
* - Asserts update mutation API is called.
* - Upon update mutation API being called, cache is invalidated and asserts cache invalidation APIS are called
*/
it("useUpdateVariable() invalidates cache and fetches updated value", async () => {
const queryClient = new QueryClient();
const MOCK_UPDATE_VARIABLE_ID = "1";
const MOCK_UPDATE_VARIABLE = {
id: MOCK_UPDATE_VARIABLE_ID,
name: "my-variable-updated",
value: 1234,
created: "2021-01-01T00:00:00Z",
updated: "2021-01-01T00:00:00Z",
tags: ["tag1"],
};

// ------------ Mock API requests
const mockUpdateVariableAPIs = () => {
const variableData = initialSeedVariables().map((variable) =>
variable.id === MOCK_UPDATE_VARIABLE_ID
? MOCK_UPDATE_VARIABLE
: variable,
);
server.use(
http.patch("http://localhost:4200/api/variables/:id", () => {
return HttpResponse.json();
}),
http.post("http://localhost:4200/api/variables/filter", () => {
return HttpResponse.json(variableData);
}),
http.post("http://localhost:4200/api/variables/count", () => {
return HttpResponse.json(variableData.length);
}),
);
};
mockUpdateVariableAPIs();

// ------------ Initialize cache
queryClient.setQueryData(
buildVariablesQuery(variableFilter).queryKey,
initialSeedVariables(),
);
queryClient.setQueryData(
buildCountQuery(variableFilter).queryKey,
initialSeedVariables().length,
);

// ------------ Initialize hooks to test
const { result: useVariablesResult } = renderHook(
() => useVariables(variableFilter),
{ wrapper: createQueryWrapper({ queryClient }) },
);
const { result: useUpdateVariableResult } = renderHook(useUpdateVariable, {
wrapper: createQueryWrapper({ queryClient }),
});

// ------------ Invoke mutation
act(() =>
useUpdateVariableResult.current.updateVariable(MOCK_UPDATE_VARIABLE),
);

// ------------ Assert
await waitFor(() =>
expect(useUpdateVariableResult.current.isSuccess).toBe(true),
);
expect(useVariablesResult.current.variables.length).toEqual(1);
expect(useVariablesResult.current.totalCount).toEqual(1);
const updatedVariable = useVariablesResult.current.variables.find(
(variable) => variable.id === MOCK_UPDATE_VARIABLE_ID,
);
expect(updatedVariable).toMatchObject(MOCK_UPDATE_VARIABLE);
});
});
4 changes: 2 additions & 2 deletions ui-v2/src/hooks/variables.ts
Original file line number Diff line number Diff line change
Expand Up @@ -52,7 +52,7 @@ const variableKeys: VariableKeys = {
* - staleTime: How long the data should be considered fresh (1 second)
* - placeholderData: Uses previous data while loading new data
*/
const buildVariablesQuery = (options: UseVariablesOptions) =>
export const buildVariablesQuery = (options: UseVariablesOptions) =>
queryOptions({
queryKey: variableKeys.filtered(options),
queryFn: async () => {
Expand All @@ -74,7 +74,7 @@ const buildVariablesQuery = (options: UseVariablesOptions) =>
* - staleTime: How long the data should be considered fresh (1 second)
* - placeholderData: Uses previous data while loading new data
*/
const buildCountQuery = (options?: UseVariablesOptions) =>
export const buildCountQuery = (options?: UseVariablesOptions) =>
queryOptions({
queryKey: variableKeys.filteredCount(options),
queryFn: async () => {
Expand Down

0 comments on commit 475a68f

Please sign in to comment.