mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
* Relax validator for installing charts Signed-off-by: Janne Savolainen <janne.savolainen@live.fi> * Tweak spacing between words in confirmation dialog Signed-off-by: Janne Savolainen <janne.savolainen@live.fi> * Add mocks for monaco editor and virtualized auto sizer to allow components to be rendered in unit tests Signed-off-by: Janne Savolainen <janne.savolainen@live.fi> * Improve typing for a function Signed-off-by: Janne Savolainen <janne.savolainen@live.fi> * Remove usage of shared global state from a component Signed-off-by: Janne Savolainen <janne.savolainen@live.fi> * Provide a way to unit test usages of storages Signed-off-by: Janne Savolainen <janne.savolainen@live.fi> * Add way to get current value from select in behavioural unit tests Signed-off-by: Janne Savolainen <janne.savolainen@live.fi> * Rework installation of helm charts to get rid of the majority of bugs Signed-off-by: Janne Savolainen <janne.savolainen@live.fi> * Update snapshots Signed-off-by: Janne Savolainen <janne.savolainen@live.fi> * Remove technical test for being covered in behaviours Signed-off-by: Janne Savolainen <janne.savolainen@live.fi> * Split behaviour to smaller pieces Signed-off-by: Janne Savolainen <janne.savolainen@live.fi> * Add tests accidentally removed back Signed-off-by: Janne Savolainen <janne.savolainen@live.fi> * Mark functions causing side effects Signed-off-by: Janne Savolainen <janne.savolainen@live.fi> * Remove behaviour covered by other behaviours Signed-off-by: Janne Savolainen <janne.savolainen@live.fi> * Tweak naming Signed-off-by: Janne Savolainen <janne.savolainen@live.fi> * Remove unused dependency Signed-off-by: Janne Savolainen <janne.savolainen@live.fi>
461 lines
15 KiB
TypeScript
461 lines
15 KiB
TypeScript
/**
|
|
* Copyright (c) OpenLens Authors. All rights reserved.
|
|
* Licensed under MIT License. See LICENSE in root directory for more information.
|
|
*/
|
|
import React from "react";
|
|
import type { RenderResult } from "@testing-library/react";
|
|
import { fireEvent } from "@testing-library/react";
|
|
import directoryForLensLocalStorageInjectable from "../../common/directory-for-lens-local-storage/directory-for-lens-local-storage.injectable";
|
|
import routesInjectable from "../../renderer/routes/routes.injectable";
|
|
import { matches } from "lodash/fp";
|
|
import type { ApplicationBuilder } from "../../renderer/components/test-utils/get-application-builder";
|
|
import { getApplicationBuilder } from "../../renderer/components/test-utils/get-application-builder";
|
|
import writeJsonFileInjectable from "../../common/fs/write-json-file.injectable";
|
|
import pathExistsInjectable from "../../common/fs/path-exists.injectable";
|
|
import readJsonFileInjectable from "../../common/fs/read-json-file.injectable";
|
|
import type { DiContainer } from "@ogre-tools/injectable";
|
|
import { navigateToRouteInjectionToken } from "../../common/front-end-routing/navigate-to-route-injection-token";
|
|
import assert from "assert";
|
|
import hostedClusterIdInjectable from "../../renderer/cluster-frame-context/hosted-cluster-id.injectable";
|
|
import { advanceFakeTime, useFakeTime } from "../../common/test-utils/use-fake-time";
|
|
import { getExtensionFakeFor } from "../../renderer/components/test-utils/get-extension-fake";
|
|
import type { IObservableValue } from "mobx";
|
|
import { runInAction, computed, observable } from "mobx";
|
|
import storageSaveDelayInjectable from "../../renderer/utils/create-storage/storage-save-delay.injectable";
|
|
|
|
// TODO: Make tooltips free of side effects by making it deterministic
|
|
jest.mock("../../renderer/components/tooltip/withTooltip", () => ({
|
|
withTooltip: (target: any) => target,
|
|
}));
|
|
|
|
describe("cluster - sidebar and tab navigation for extensions", () => {
|
|
let applicationBuilder: ApplicationBuilder;
|
|
let rendererDi: DiContainer;
|
|
let rendered: RenderResult;
|
|
|
|
beforeEach(() => {
|
|
useFakeTime("2015-10-21T07:28:00Z");
|
|
|
|
applicationBuilder = getApplicationBuilder();
|
|
rendererDi = applicationBuilder.dis.rendererDi;
|
|
|
|
applicationBuilder.setEnvironmentToClusterFrame();
|
|
|
|
applicationBuilder.beforeApplicationStart(({ rendererDi }) => {
|
|
rendererDi.override(hostedClusterIdInjectable, () => "some-hosted-cluster-id");
|
|
|
|
rendererDi.override(storageSaveDelayInjectable, () => 250);
|
|
|
|
rendererDi.override(
|
|
directoryForLensLocalStorageInjectable,
|
|
() => "/some-directory-for-lens-local-storage",
|
|
);
|
|
});
|
|
});
|
|
|
|
describe("given extension with cluster pages and cluster page menus", () => {
|
|
let someObservable: IObservableValue<boolean>;
|
|
|
|
beforeEach(() => {
|
|
someObservable = observable.box(false);
|
|
|
|
const getExtensionFake = getExtensionFakeFor(applicationBuilder);
|
|
|
|
const testExtension = getExtensionFake({
|
|
id: "some-extension-id",
|
|
name: "some-extension-name",
|
|
|
|
rendererOptions: {
|
|
clusterPages: [
|
|
{
|
|
components: {
|
|
Page: () => {
|
|
throw new Error("should never come here");
|
|
},
|
|
},
|
|
},
|
|
{
|
|
id: "some-child-page-id",
|
|
|
|
components: {
|
|
Page: () => <div data-testid="some-child-page">Some child page</div>,
|
|
},
|
|
},
|
|
{
|
|
id: "some-other-child-page-id",
|
|
|
|
components: {
|
|
Page: () => (
|
|
<div data-testid="some-other-child-page">Some other child page</div>
|
|
),
|
|
},
|
|
},
|
|
],
|
|
|
|
clusterPageMenus: [
|
|
{
|
|
id: "some-parent-id",
|
|
title: "Parent",
|
|
|
|
components: {
|
|
Icon: () => <div>Some icon</div>,
|
|
},
|
|
},
|
|
|
|
{
|
|
id: "some-child-id",
|
|
target: { pageId: "some-child-page-id" },
|
|
parentId: "some-parent-id",
|
|
title: "Child 1",
|
|
|
|
components: {
|
|
Icon: null as never,
|
|
},
|
|
},
|
|
|
|
{
|
|
id: "some-other-child-id",
|
|
target: { pageId: "some-other-child-page-id" },
|
|
parentId: "some-parent-id",
|
|
title: "Child 2",
|
|
|
|
components: {
|
|
Icon: null as never,
|
|
},
|
|
},
|
|
|
|
{
|
|
id: "some-menu-with-controlled-visibility",
|
|
title: "Some menu with controlled visibility",
|
|
visible: computed(() => someObservable.get()),
|
|
|
|
components: {
|
|
Icon: () => <div>Some icon</div>,
|
|
},
|
|
},
|
|
],
|
|
},
|
|
});
|
|
|
|
applicationBuilder.extensions.enable(testExtension);
|
|
});
|
|
|
|
describe("given no state for expanded sidebar items exists, and navigated to child sidebar item, when rendered", () => {
|
|
beforeEach(async () => {
|
|
applicationBuilder.beforeRender(({ rendererDi }) => {
|
|
const navigateToRoute = rendererDi.inject(navigateToRouteInjectionToken);
|
|
const route = rendererDi
|
|
.inject(routesInjectable)
|
|
.get()
|
|
.find(
|
|
matches({
|
|
path: "/extension/some-extension-name/some-child-page-id",
|
|
}),
|
|
);
|
|
|
|
assert(route);
|
|
navigateToRoute(route);
|
|
});
|
|
|
|
rendered = await applicationBuilder.render();
|
|
});
|
|
|
|
it("renders", () => {
|
|
expect(rendered.container).toMatchSnapshot();
|
|
});
|
|
|
|
it("parent is highlighted", () => {
|
|
const parent = rendered.getByTestId("sidebar-item-some-extension-name-some-parent-id");
|
|
|
|
expect(parent?.dataset.isActiveTest).toBe("true");
|
|
});
|
|
|
|
it("parent sidebar item is not expanded", () => {
|
|
const child = rendered.queryByTestId("sidebar-item-some-extension-name-some-child-id");
|
|
|
|
expect(child).toBeNull();
|
|
});
|
|
|
|
it("child page is shown", () => {
|
|
expect(rendered.getByTestId("some-child-page")).not.toBeNull();
|
|
});
|
|
});
|
|
|
|
describe("given state for expanded sidebar items already exists, when rendered", () => {
|
|
beforeEach(async () => {
|
|
applicationBuilder.beforeRender(async ({ rendererDi }) => {
|
|
const writeJsonFileFake = rendererDi.inject(writeJsonFileInjectable);
|
|
|
|
await writeJsonFileFake(
|
|
"/some-directory-for-lens-local-storage/some-hosted-cluster-id.json",
|
|
{
|
|
sidebar: {
|
|
expanded: { "some-extension-name-some-parent-id": true },
|
|
width: 200,
|
|
},
|
|
},
|
|
);
|
|
});
|
|
|
|
rendered = await applicationBuilder.render();
|
|
});
|
|
|
|
it("renders", () => {
|
|
expect(rendered.container).toMatchSnapshot();
|
|
});
|
|
|
|
it("parent sidebar item is not highlighted", () => {
|
|
const parent = rendered.getByTestId("sidebar-item-some-extension-name-some-parent-id");
|
|
|
|
expect(parent?.dataset.isActiveTest).toBe("false");
|
|
});
|
|
|
|
it("parent sidebar item is expanded", () => {
|
|
const child = rendered.queryByTestId("sidebar-item-some-extension-name-some-child-id");
|
|
|
|
expect(child).not.toBeNull();
|
|
});
|
|
});
|
|
|
|
describe("given state for expanded unknown sidebar items already exists, when rendered", () => {
|
|
beforeEach(async () => {
|
|
applicationBuilder.beforeRender(async ({ rendererDi }) => {
|
|
const writeJsonFileFake = rendererDi.inject(writeJsonFileInjectable);
|
|
|
|
await writeJsonFileFake(
|
|
"/some-directory-for-lens-local-storage/some-hosted-cluster-id.json",
|
|
{
|
|
sidebar: {
|
|
expanded: { "some-extension-name-some-unknown-parent-id": true },
|
|
width: 200,
|
|
},
|
|
},
|
|
);
|
|
});
|
|
|
|
rendered = await applicationBuilder.render();
|
|
});
|
|
|
|
it("renders without errors", () => {
|
|
expect(rendered.container).toMatchSnapshot();
|
|
});
|
|
|
|
it("parent sidebar item is not expanded", () => {
|
|
const child = rendered.queryByTestId("sidebar-item-some-extension-name-some-child-id");
|
|
|
|
expect(child).toBeNull();
|
|
});
|
|
});
|
|
|
|
describe("given empty state for expanded sidebar items already exists, when rendered", () => {
|
|
beforeEach(async () => {
|
|
applicationBuilder.beforeRender(async ({ rendererDi }) => {
|
|
const writeJsonFileFake = rendererDi.inject(writeJsonFileInjectable);
|
|
|
|
await writeJsonFileFake(
|
|
"/some-directory-for-lens-local-storage/some-hosted-cluster-id.json",
|
|
{
|
|
someThingButSidebar: {},
|
|
},
|
|
);
|
|
});
|
|
|
|
rendered = await applicationBuilder.render();
|
|
});
|
|
|
|
it("renders without errors", () => {
|
|
expect(rendered.container).toMatchSnapshot();
|
|
});
|
|
|
|
it("parent sidebar item is not expanded", () => {
|
|
const child = rendered.queryByTestId("sidebar-item-some-extension-name-some-child-id");
|
|
|
|
expect(child).toBeNull();
|
|
});
|
|
});
|
|
|
|
describe("given no initially persisted state for sidebar items, when rendered", () => {
|
|
beforeEach(async () => {
|
|
rendered = await applicationBuilder.render();
|
|
});
|
|
|
|
it("renders", () => {
|
|
expect(rendered.container).toMatchSnapshot();
|
|
});
|
|
|
|
it("parent sidebar item is not highlighted", () => {
|
|
const parent = rendered.getByTestId("sidebar-item-some-extension-name-some-parent-id");
|
|
|
|
expect(parent?.dataset.isActiveTest).toBe("false");
|
|
});
|
|
|
|
it("parent sidebar item is not expanded", () => {
|
|
const child = rendered.queryByTestId("sidebar-item-some-extension-name-some-child-id");
|
|
|
|
expect(child).toBeNull();
|
|
});
|
|
|
|
it("does not show the sidebar item that should be hidden", () => {
|
|
const child = rendered.queryByTestId(
|
|
"sidebar-item-some-extension-name-some-menu-with-controlled-visibility",
|
|
);
|
|
|
|
expect(child).not.toBeInTheDocument();
|
|
});
|
|
|
|
it("when sidebar item becomes visible, shows the sidebar item", () => {
|
|
runInAction(() => {
|
|
someObservable.set(true);
|
|
});
|
|
|
|
const child = rendered.queryByTestId(
|
|
"sidebar-item-some-extension-name-some-menu-with-controlled-visibility",
|
|
);
|
|
|
|
expect(child).toBeInTheDocument();
|
|
});
|
|
|
|
describe("when a parent sidebar item is expanded", () => {
|
|
beforeEach(() => {
|
|
const parentLink = rendered.getByTestId(
|
|
"sidebar-item-link-for-some-extension-name-some-parent-id",
|
|
);
|
|
|
|
fireEvent.click(parentLink);
|
|
});
|
|
|
|
it("renders", () => {
|
|
expect(rendered.container).toMatchSnapshot();
|
|
});
|
|
|
|
it("parent sidebar item is not highlighted", () => {
|
|
const parent = rendered.getByTestId("sidebar-item-some-extension-name-some-parent-id");
|
|
|
|
expect(parent?.dataset.isActiveTest).toBe("false");
|
|
});
|
|
|
|
it("parent sidebar item is expanded", () => {
|
|
const child = rendered.queryByTestId("sidebar-item-some-extension-name-some-child-id");
|
|
|
|
expect(child).not.toBeNull();
|
|
});
|
|
|
|
describe("when a child of the parent is selected", () => {
|
|
beforeEach(() => {
|
|
const childLink = rendered.getByTestId(
|
|
"sidebar-item-link-for-some-extension-name-some-child-id",
|
|
);
|
|
|
|
fireEvent.click(childLink);
|
|
});
|
|
|
|
it("renders", () => {
|
|
expect(rendered.container).toMatchSnapshot();
|
|
});
|
|
|
|
it("parent is highlighted", () => {
|
|
const parent = rendered.getByTestId("sidebar-item-some-extension-name-some-parent-id");
|
|
|
|
expect(parent?.dataset.isActiveTest).toBe("true");
|
|
});
|
|
|
|
it("child is highlighted", () => {
|
|
const child = rendered.getByTestId("sidebar-item-some-extension-name-some-child-id");
|
|
|
|
expect(child?.dataset.isActiveTest).toBe("true");
|
|
});
|
|
|
|
it("child page is shown", () => {
|
|
expect(rendered.getByTestId("some-child-page")).not.toBeNull();
|
|
});
|
|
|
|
it("renders tabs", () => {
|
|
expect(rendered.getByTestId("tab-layout")).not.toBeNull();
|
|
});
|
|
|
|
it("tab for child page is active", () => {
|
|
const tabLink = rendered.getByTestId(
|
|
"tab-link-for-some-extension-name-some-child-id",
|
|
);
|
|
|
|
expect(tabLink.dataset.isActiveTest).toBe("true");
|
|
});
|
|
|
|
it("tab for sibling page is not active", () => {
|
|
const tabLink = rendered.getByTestId(
|
|
"tab-link-for-some-extension-name-some-other-child-id",
|
|
);
|
|
|
|
expect(tabLink.dataset.isActiveTest).toBe("false");
|
|
});
|
|
|
|
it("when not enough time passes, does not store state for expanded sidebar items to file system yet", async () => {
|
|
advanceFakeTime(250 - 1);
|
|
|
|
const pathExistsFake = rendererDi.inject(pathExistsInjectable);
|
|
|
|
const actual = await pathExistsFake(
|
|
"/some-directory-for-lens-local-storage/some-hosted-cluster-id.json",
|
|
);
|
|
|
|
expect(actual).toBe(false);
|
|
});
|
|
|
|
it("when enough time passes, stores state for expanded sidebar items to file system", async () => {
|
|
advanceFakeTime(250);
|
|
|
|
const readJsonFileFake = rendererDi.inject(readJsonFileInjectable);
|
|
|
|
const actual = await readJsonFileFake(
|
|
"/some-directory-for-lens-local-storage/some-hosted-cluster-id.json",
|
|
);
|
|
|
|
expect(actual).toEqual({
|
|
sidebar: {
|
|
expanded: { "some-extension-name-some-parent-id": true },
|
|
width: 200,
|
|
},
|
|
});
|
|
});
|
|
|
|
describe("when selecting sibling tab", () => {
|
|
beforeEach(() => {
|
|
const childTabLink = rendered.getByTestId(
|
|
"tab-link-for-some-extension-name-some-other-child-id",
|
|
);
|
|
|
|
fireEvent.click(childTabLink);
|
|
});
|
|
|
|
it("renders", () => {
|
|
expect(rendered.container).toMatchSnapshot();
|
|
});
|
|
|
|
it("sibling child page is shown", () => {
|
|
expect(
|
|
rendered.getByTestId("some-other-child-page"),
|
|
).not.toBeNull();
|
|
});
|
|
|
|
it("tab for sibling page is active", () => {
|
|
const tabLink = rendered.getByTestId(
|
|
"tab-link-for-some-extension-name-some-other-child-id",
|
|
);
|
|
|
|
expect(tabLink.dataset.isActiveTest).toBe("true");
|
|
});
|
|
|
|
it("tab for previous page is not active", () => {
|
|
const tabLink = rendered.getByTestId(
|
|
"tab-link-for-some-extension-name-some-child-id",
|
|
);
|
|
|
|
expect(tabLink.dataset.isActiveTest).toBe("false");
|
|
});
|
|
});
|
|
});
|
|
});
|
|
});
|
|
});
|
|
});
|