coder/site/src/pages/CreateTemplatePage/CreateTemplatePage.test.tsx

145 lines
5.1 KiB
TypeScript

import { renderWithAuth } from "testHelpers/renderHelpers";
import CreateTemplatePage from "./CreateTemplatePage";
import { screen, waitFor, within } from "@testing-library/react";
import userEvent from "@testing-library/user-event";
import * as API from "api/api";
import {
MockTemplateExample,
MockTemplateVersion,
MockTemplateVersionVariable1,
MockTemplateVersionVariable2,
MockTemplateVersionVariable3,
MockTemplate,
MockOrganization,
} from "testHelpers/entities";
const renderPage = async (searchParams: URLSearchParams) => {
// Render with the example ID so we don't need to upload a file
const view = renderWithAuth(<CreateTemplatePage />, {
route: `/templates/new?${searchParams.toString()}`,
path: "/templates/new",
// We need this because after creation, the user will be redirected to here
extraRoutes: [{ path: "templates/:template", element: <></> }],
});
// It is lazy loaded, so we have to wait for it to be rendered to not get an
// act error
await screen.findByLabelText("Icon", undefined, { timeout: 5000 });
return view;
};
test("Create template from starter template", async () => {
// Render page, fill the name and submit
const searchParams = new URLSearchParams({
exampleId: MockTemplateExample.id,
});
const { router, container } = await renderPage(searchParams);
const form = container.querySelector("form") as HTMLFormElement;
jest.spyOn(API, "createTemplateVersion").mockResolvedValueOnce({
...MockTemplateVersion,
job: {
...MockTemplateVersion.job,
status: "pending",
},
});
jest.spyOn(API, "getTemplateVersion").mockResolvedValue({
...MockTemplateVersion,
job: {
...MockTemplateVersion.job,
status: "failed",
error_code: "REQUIRED_TEMPLATE_VARIABLES",
},
});
jest
.spyOn(API, "getTemplateVersionVariables")
.mockResolvedValue([
MockTemplateVersionVariable1,
MockTemplateVersionVariable2,
MockTemplateVersionVariable3,
]);
await userEvent.type(screen.getByLabelText(/Name/), "my-template");
await userEvent.click(
within(form).getByRole("button", { name: /create template/i }),
);
// Wait for the variables form to be rendered and fill it
await screen.findByText(/Variables/, undefined, { timeout: 5_000 });
// Type first variable
await userEvent.clear(screen.getByLabelText(/var.first_variable/));
await userEvent.type(
screen.getByLabelText(/var.first_variable/),
"First value",
);
// Type second variable
await userEvent.clear(screen.getByLabelText(/var.second_variable/));
await userEvent.type(screen.getByLabelText(/var.second_variable/), "2");
// Select third variable on radio
await userEvent.click(screen.getByLabelText(/True/));
// Setup the mock for the second template version creation before submit the form
jest.clearAllMocks();
jest
.spyOn(API, "createTemplateVersion")
.mockResolvedValue(MockTemplateVersion);
jest.spyOn(API, "getTemplateVersion").mockResolvedValue(MockTemplateVersion);
jest.spyOn(API, "createTemplate").mockResolvedValue(MockTemplate);
await userEvent.click(
within(form).getByRole("button", { name: /create template/i }),
);
await waitFor(() => expect(API.createTemplate).toBeCalledTimes(1));
expect(router.state.location.pathname).toEqual(
`/templates/${MockTemplate.name}/files`,
);
expect(API.createTemplateVersion).toHaveBeenCalledWith(MockOrganization.id, {
example_id: "aws-windows",
provisioner: "terraform",
storage_method: "file",
tags: {},
user_variable_values: [
{ name: "first_variable", value: "First value" },
{ name: "second_variable", value: "2" },
{ name: "third_variable", value: "true" },
],
});
});
test("Create template from duplicating a template", async () => {
jest.spyOn(API, "getTemplateByName").mockResolvedValue(MockTemplate);
jest.spyOn(API, "getTemplateVersion").mockResolvedValue(MockTemplateVersion);
jest
.spyOn(API, "getTemplateVersionVariables")
.mockResolvedValue([MockTemplateVersionVariable1]);
const searchParams = new URLSearchParams({
fromTemplate: MockTemplate.name,
});
const { router } = await renderPage(searchParams);
// Name and display name are using copy prefixes
expect(screen.getByLabelText(/Name/)).toHaveValue(
`${MockTemplate.name}-copy`,
);
expect(screen.getByLabelText(/Display name/)).toHaveValue(
`Copy of ${MockTemplate.display_name}`,
);
// Variables are using the same values
expect(
screen.getByLabelText(MockTemplateVersionVariable1.description, {
exact: false,
}),
).toHaveValue(MockTemplateVersionVariable1.value);
// Create template
jest
.spyOn(API, "createTemplateVersion")
.mockResolvedValue(MockTemplateVersion);
jest.spyOn(API, "getTemplateVersion").mockResolvedValue(MockTemplateVersion);
jest.spyOn(API, "createTemplate").mockResolvedValue(MockTemplate);
await userEvent.click(
screen.getByRole("button", { name: /create template/i }),
);
await waitFor(() => {
expect(router.state.location.pathname).toEqual(
`/templates/${MockTemplate.name}/files`,
);
});
});