mirror of https://github.com/coder/coder.git
277 lines
8.5 KiB
TypeScript
277 lines
8.5 KiB
TypeScript
import { fireEvent, screen, waitFor } from "@testing-library/react";
|
|
import userEvent from "@testing-library/user-event";
|
|
import * as API from "api/api";
|
|
import {
|
|
MockTemplate,
|
|
MockUser,
|
|
MockWorkspace,
|
|
MockWorkspaceQuota,
|
|
MockWorkspaceRichParametersRequest,
|
|
MockTemplateVersionParameter1,
|
|
MockTemplateVersionParameter2,
|
|
MockTemplateVersionParameter3,
|
|
MockTemplateVersionExternalAuthGithub,
|
|
MockOrganization,
|
|
} from "testHelpers/entities";
|
|
import {
|
|
renderWithAuth,
|
|
waitForLoaderToBeRemoved,
|
|
} from "testHelpers/renderHelpers";
|
|
import CreateWorkspacePage from "./CreateWorkspacePage";
|
|
import { Language } from "./CreateWorkspacePageView";
|
|
import { server } from "testHelpers/server";
|
|
import { rest } from "msw";
|
|
|
|
const nameLabelText = "Workspace Name";
|
|
const createWorkspaceText = "Create Workspace";
|
|
const validationNumberNotInRangeText = "Value must be between 1 and 3.";
|
|
const validationPatternNotMatched = `${MockTemplateVersionParameter3.validation_error} (value does not match the pattern ^[a-z]{3}$)`;
|
|
|
|
const renderCreateWorkspacePage = () => {
|
|
return renderWithAuth(<CreateWorkspacePage />, {
|
|
route: "/templates/" + MockTemplate.name + "/workspace",
|
|
path: "/templates/:template/workspace",
|
|
});
|
|
};
|
|
|
|
describe("CreateWorkspacePage", () => {
|
|
it("succeeds with default owner", async () => {
|
|
jest
|
|
.spyOn(API, "getUsers")
|
|
.mockResolvedValueOnce({ users: [MockUser], count: 1 });
|
|
jest
|
|
.spyOn(API, "getWorkspaceQuota")
|
|
.mockResolvedValueOnce(MockWorkspaceQuota);
|
|
jest.spyOn(API, "createWorkspace").mockResolvedValueOnce(MockWorkspace);
|
|
jest
|
|
.spyOn(API, "getTemplateVersionRichParameters")
|
|
.mockResolvedValueOnce([MockTemplateVersionParameter1]);
|
|
|
|
renderCreateWorkspacePage();
|
|
|
|
const nameField = await screen.findByLabelText(nameLabelText);
|
|
|
|
// have to use fireEvent b/c userEvent isn't cleaning up properly between tests
|
|
fireEvent.change(nameField, {
|
|
target: { value: "test" },
|
|
});
|
|
|
|
const submitButton = screen.getByText(createWorkspaceText);
|
|
await userEvent.click(submitButton);
|
|
|
|
await waitFor(() =>
|
|
expect(API.createWorkspace).toBeCalledWith(
|
|
MockUser.organization_ids[0],
|
|
MockUser.id,
|
|
expect.objectContaining({
|
|
...MockWorkspaceRichParametersRequest,
|
|
}),
|
|
),
|
|
);
|
|
});
|
|
|
|
it("uses default rich param values passed from the URL", async () => {
|
|
const param = "first_parameter";
|
|
const paramValue = "It works!";
|
|
jest
|
|
.spyOn(API, "getTemplateVersionRichParameters")
|
|
.mockResolvedValueOnce([MockTemplateVersionParameter1]);
|
|
|
|
renderWithAuth(<CreateWorkspacePage />, {
|
|
route:
|
|
"/templates/" +
|
|
MockTemplate.name +
|
|
`/workspace?param.${param}=${paramValue}`,
|
|
path: "/templates/:template/workspace",
|
|
});
|
|
|
|
await screen.findByDisplayValue(paramValue);
|
|
});
|
|
|
|
it("rich parameter: number validation fails", async () => {
|
|
jest
|
|
.spyOn(API, "getTemplateVersionRichParameters")
|
|
.mockResolvedValueOnce([
|
|
MockTemplateVersionParameter1,
|
|
MockTemplateVersionParameter2,
|
|
]);
|
|
|
|
renderCreateWorkspacePage();
|
|
await waitForLoaderToBeRemoved();
|
|
|
|
const element = await screen.findByText("Create Workspace");
|
|
expect(element).toBeDefined();
|
|
const secondParameter = await screen.findByText(
|
|
MockTemplateVersionParameter2.description,
|
|
);
|
|
expect(secondParameter).toBeDefined();
|
|
|
|
const secondParameterField = await screen.findByLabelText(
|
|
MockTemplateVersionParameter2.name,
|
|
{ exact: false },
|
|
);
|
|
expect(secondParameterField).toBeDefined();
|
|
|
|
fireEvent.change(secondParameterField, {
|
|
target: { value: "4" },
|
|
});
|
|
fireEvent.submit(secondParameter);
|
|
|
|
const validationError = await screen.findByText(
|
|
validationNumberNotInRangeText,
|
|
);
|
|
expect(validationError).toBeDefined();
|
|
});
|
|
|
|
it("rich parameter: string validation fails", async () => {
|
|
jest
|
|
.spyOn(API, "getTemplateVersionRichParameters")
|
|
.mockResolvedValueOnce([
|
|
MockTemplateVersionParameter1,
|
|
MockTemplateVersionParameter3,
|
|
]);
|
|
|
|
renderCreateWorkspacePage();
|
|
await waitForLoaderToBeRemoved();
|
|
|
|
const element = await screen.findByText(createWorkspaceText);
|
|
expect(element).toBeDefined();
|
|
const thirdParameter = await screen.findByText(
|
|
MockTemplateVersionParameter3.description,
|
|
);
|
|
expect(thirdParameter).toBeDefined();
|
|
|
|
const thirdParameterField = await screen.findByLabelText(
|
|
MockTemplateVersionParameter3.name,
|
|
{ exact: false },
|
|
);
|
|
expect(thirdParameterField).toBeDefined();
|
|
fireEvent.change(thirdParameterField, {
|
|
target: { value: "1234" },
|
|
});
|
|
fireEvent.submit(thirdParameterField);
|
|
|
|
const validationError = await screen.findByText(
|
|
validationPatternNotMatched,
|
|
);
|
|
expect(validationError).toBeInTheDocument();
|
|
});
|
|
|
|
it("auto create a workspace if uses mode=auto", async () => {
|
|
const param = "first_parameter";
|
|
const paramValue = "It works!";
|
|
const createWorkspaceSpy = jest.spyOn(API, "createWorkspace");
|
|
|
|
renderWithAuth(<CreateWorkspacePage />, {
|
|
route:
|
|
"/templates/" +
|
|
MockTemplate.name +
|
|
`/workspace?param.${param}=${paramValue}&mode=auto`,
|
|
path: "/templates/:template/workspace",
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(createWorkspaceSpy).toBeCalledWith(
|
|
MockOrganization.id,
|
|
"me",
|
|
expect.objectContaining({
|
|
template_id: MockTemplate.id,
|
|
rich_parameter_values: [
|
|
expect.objectContaining({ name: param, value: paramValue }),
|
|
],
|
|
}),
|
|
);
|
|
});
|
|
});
|
|
|
|
it("auto create a workspace if uses mode=auto and version=version-id", async () => {
|
|
const param = "first_parameter";
|
|
const paramValue = "It works!";
|
|
const createWorkspaceSpy = jest.spyOn(API, "createWorkspace");
|
|
|
|
renderWithAuth(<CreateWorkspacePage />, {
|
|
route:
|
|
"/templates/" +
|
|
MockTemplate.name +
|
|
`/workspace?param.${param}=${paramValue}&mode=auto&version=test-template-version`,
|
|
path: "/templates/:template/workspace",
|
|
});
|
|
|
|
await waitFor(() => {
|
|
expect(createWorkspaceSpy).toBeCalledWith(
|
|
MockOrganization.id,
|
|
"me",
|
|
expect.objectContaining({
|
|
template_version_id: MockTemplate.active_version_id,
|
|
rich_parameter_values: [
|
|
expect.objectContaining({ name: param, value: paramValue }),
|
|
],
|
|
}),
|
|
);
|
|
});
|
|
});
|
|
|
|
it("Detects when a workspace is being created with the 'duplicate' mode", async () => {
|
|
const params = new URLSearchParams({
|
|
mode: "duplicate",
|
|
name: MockWorkspace.name,
|
|
version: MockWorkspace.template_active_version_id,
|
|
});
|
|
|
|
renderWithAuth(<CreateWorkspacePage />, {
|
|
path: "/templates/:template/workspace",
|
|
route: `/templates/${MockWorkspace.name}/workspace?${params.toString()}`,
|
|
});
|
|
|
|
const warningMessage = await screen.findByRole("alert");
|
|
const nameInput = await screen.findByRole("textbox", {
|
|
name: "Workspace Name",
|
|
});
|
|
|
|
expect(warningMessage).toHaveTextContent(Language.duplicationWarning);
|
|
expect(nameInput).toHaveValue(`${MockWorkspace.name}-copy`);
|
|
});
|
|
|
|
it("displays the form after connecting to all the external services", async () => {
|
|
jest.spyOn(window, "open").mockImplementation(() => null);
|
|
const user = userEvent.setup();
|
|
const notAuthenticatedExternalAuth = {
|
|
...MockTemplateVersionExternalAuthGithub,
|
|
authenticated: false,
|
|
};
|
|
server.use(
|
|
rest.get(
|
|
"/api/v2/templateversions/:versionId/external-auth",
|
|
(req, res, ctx) => {
|
|
return res(ctx.json([notAuthenticatedExternalAuth]));
|
|
},
|
|
),
|
|
);
|
|
renderCreateWorkspacePage();
|
|
|
|
await screen.findByText("External authentication");
|
|
expect(screen.queryByRole("form")).not.toBeInTheDocument();
|
|
|
|
const connectButton = screen.getByRole("button", {
|
|
name: /connect/i,
|
|
});
|
|
server.use(
|
|
rest.get(
|
|
"/api/v2/templateversions/:versionId/external-auth",
|
|
(req, res, ctx) => {
|
|
const authenticatedExternalAuth = {
|
|
...MockTemplateVersionExternalAuthGithub,
|
|
authenticated: true,
|
|
};
|
|
return res(ctx.json([authenticatedExternalAuth]));
|
|
},
|
|
),
|
|
);
|
|
await user.click(connectButton);
|
|
// TODO: Consider improving the timeout by simulating react-query polling.
|
|
// Current implementation could not achieve this, further research is
|
|
// needed.
|
|
await screen.findByRole("form", undefined, { timeout: 10_000 });
|
|
});
|
|
});
|