mirror of
https://github.com/scm-manager/scm-manager.git
synced 2025-11-09 23:15:43 +01:00
318 lines
13 KiB
TypeScript
318 lines
13 KiB
TypeScript
|
|
/*
|
||
|
|
* MIT License
|
||
|
|
*
|
||
|
|
* Copyright (c) 2020-present Cloudogu GmbH and Contributors
|
||
|
|
*
|
||
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
|
|
* of this software and associated documentation files (the "Software"), to deal
|
||
|
|
* in the Software without restriction, including without limitation the rights
|
||
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
|
|
* copies of the Software, and to permit persons to whom the Software is
|
||
|
|
* furnished to do so, subject to the following conditions:
|
||
|
|
*
|
||
|
|
* The above copyright notice and this permission notice shall be included in all
|
||
|
|
* copies or substantial portions of the Software.
|
||
|
|
*
|
||
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||
|
|
* SOFTWARE.
|
||
|
|
*/
|
||
|
|
|
||
|
|
import { PendingPlugins, Plugin, PluginCollection } from "@scm-manager/ui-types";
|
||
|
|
import createInfiniteCachingClient from "./tests/createInfiniteCachingClient";
|
||
|
|
import { setIndexLink } from "./tests/indexLinks";
|
||
|
|
import fetchMock from "fetch-mock-jest";
|
||
|
|
import { renderHook } from "@testing-library/react-hooks";
|
||
|
|
import createWrapper from "./tests/createWrapper";
|
||
|
|
import {
|
||
|
|
useAvailablePlugins,
|
||
|
|
useInstalledPlugins,
|
||
|
|
useInstallPlugin,
|
||
|
|
usePendingPlugins,
|
||
|
|
useUninstallPlugin,
|
||
|
|
useUpdatePlugins
|
||
|
|
} from "./plugins";
|
||
|
|
import { act } from "react-test-renderer";
|
||
|
|
|
||
|
|
describe("Test plugin hooks", () => {
|
||
|
|
const availablePlugin: Plugin = {
|
||
|
|
author: "Douglas Adams",
|
||
|
|
category: "all",
|
||
|
|
displayName: "Heart of Gold",
|
||
|
|
version: "x.y.z",
|
||
|
|
name: "heart-of-gold-plugin",
|
||
|
|
pending: false,
|
||
|
|
dependencies: [],
|
||
|
|
optionalDependencies: [],
|
||
|
|
_links: {
|
||
|
|
install: { href: "/plugins/available/heart-of-gold-plugin/install" },
|
||
|
|
installWithRestart: {
|
||
|
|
href: "/plugins/available/heart-of-gold-plugin/install?restart=true"
|
||
|
|
}
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
const installedPlugin: Plugin = {
|
||
|
|
author: "Douglas Adams",
|
||
|
|
category: "all",
|
||
|
|
displayName: "Heart of Gold",
|
||
|
|
version: "x.y.z",
|
||
|
|
name: "heart-of-gold-plugin",
|
||
|
|
pending: false,
|
||
|
|
markedForUninstall: false,
|
||
|
|
dependencies: [],
|
||
|
|
optionalDependencies: [],
|
||
|
|
_links: {
|
||
|
|
self: {
|
||
|
|
href: "/plugins/installed/heart-of-gold-plugin"
|
||
|
|
},
|
||
|
|
update: {
|
||
|
|
href: "/plugins/available/heart-of-gold-plugin/install"
|
||
|
|
},
|
||
|
|
updateWithRestart: {
|
||
|
|
href: "/plugins/available/heart-of-gold-plugin/install?restart=true"
|
||
|
|
},
|
||
|
|
uninstall: {
|
||
|
|
href: "/plugins/installed/heart-of-gold-plugin/uninstall"
|
||
|
|
},
|
||
|
|
uninstallWithRestart: {
|
||
|
|
href: "/plugins/installed/heart-of-gold-plugin/uninstall?restart=true"
|
||
|
|
}
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
const installedCorePlugin: Plugin = {
|
||
|
|
author: "Douglas Adams",
|
||
|
|
category: "all",
|
||
|
|
displayName: "Heart of Gold",
|
||
|
|
version: "x.y.z",
|
||
|
|
name: "heart-of-gold-core-plugin",
|
||
|
|
pending: false,
|
||
|
|
markedForUninstall: false,
|
||
|
|
dependencies: [],
|
||
|
|
optionalDependencies: [],
|
||
|
|
_links: {
|
||
|
|
self: {
|
||
|
|
href: "/plugins/installed/heart-of-gold-core-plugin"
|
||
|
|
}
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
const createPluginCollection = (plugins: Plugin[]): PluginCollection => ({
|
||
|
|
_links: {
|
||
|
|
update: {
|
||
|
|
href: "/plugins/update"
|
||
|
|
}
|
||
|
|
},
|
||
|
|
_embedded: {
|
||
|
|
plugins
|
||
|
|
}
|
||
|
|
});
|
||
|
|
|
||
|
|
const createPendingPlugins = (
|
||
|
|
newPlugins: Plugin[] = [],
|
||
|
|
updatePlugins: Plugin[] = [],
|
||
|
|
uninstallPlugins: Plugin[] = []
|
||
|
|
): PendingPlugins => ({
|
||
|
|
_links: {},
|
||
|
|
_embedded: {
|
||
|
|
new: newPlugins,
|
||
|
|
update: updatePlugins,
|
||
|
|
uninstall: uninstallPlugins
|
||
|
|
}
|
||
|
|
});
|
||
|
|
|
||
|
|
afterEach(() => fetchMock.reset());
|
||
|
|
|
||
|
|
describe("useAvailablePlugins tests", () => {
|
||
|
|
it("should return availablePlugins", async () => {
|
||
|
|
const availablePlugins = createPluginCollection([availablePlugin]);
|
||
|
|
const queryClient = createInfiniteCachingClient();
|
||
|
|
setIndexLink(queryClient, "availablePlugins", "/availablePlugins");
|
||
|
|
fetchMock.get("/api/v2/availablePlugins", availablePlugins);
|
||
|
|
const { result, waitFor } = renderHook(() => useAvailablePlugins(), {
|
||
|
|
wrapper: createWrapper(undefined, queryClient)
|
||
|
|
});
|
||
|
|
await waitFor(() => !!result.current.data);
|
||
|
|
expect(result.current.data).toEqual(availablePlugins);
|
||
|
|
});
|
||
|
|
});
|
||
|
|
|
||
|
|
describe("useInstalledPlugins tests", () => {
|
||
|
|
it("should return installedPlugins", async () => {
|
||
|
|
const installedPlugins = createPluginCollection([installedPlugin, installedCorePlugin]);
|
||
|
|
const queryClient = createInfiniteCachingClient();
|
||
|
|
setIndexLink(queryClient, "installedPlugins", "/installedPlugins");
|
||
|
|
fetchMock.get("/api/v2/installedPlugins", installedPlugins);
|
||
|
|
const { result, waitFor } = renderHook(() => useInstalledPlugins(), {
|
||
|
|
wrapper: createWrapper(undefined, queryClient)
|
||
|
|
});
|
||
|
|
await waitFor(() => !!result.current.data);
|
||
|
|
expect(result.current.data).toEqual(installedPlugins);
|
||
|
|
});
|
||
|
|
});
|
||
|
|
|
||
|
|
describe("usePendingPlugins tests", () => {
|
||
|
|
it("should return pendingPlugins", async () => {
|
||
|
|
const pendingPlugins = createPendingPlugins([availablePlugin]);
|
||
|
|
const queryClient = createInfiniteCachingClient();
|
||
|
|
setIndexLink(queryClient, "pendingPlugins", "/pendingPlugins");
|
||
|
|
fetchMock.get("/api/v2/pendingPlugins", pendingPlugins);
|
||
|
|
const { result, waitFor } = renderHook(() => usePendingPlugins(), {
|
||
|
|
wrapper: createWrapper(undefined, queryClient)
|
||
|
|
});
|
||
|
|
await waitFor(() => !!result.current.data);
|
||
|
|
expect(result.current.data).toEqual(pendingPlugins);
|
||
|
|
});
|
||
|
|
});
|
||
|
|
|
||
|
|
describe("useInstallPlugin tests", () => {
|
||
|
|
it("should use restart parameter", async () => {
|
||
|
|
const queryClient = createInfiniteCachingClient();
|
||
|
|
queryClient.setQueryData(["plugins", "available"], createPluginCollection([availablePlugin]));
|
||
|
|
queryClient.setQueryData(["plugins", "installed"], createPluginCollection([]));
|
||
|
|
queryClient.setQueryData(["plugins", "pending"], createPendingPlugins());
|
||
|
|
fetchMock.post("/api/v2/plugins/available/heart-of-gold-plugin/install?restart=true", installedPlugin);
|
||
|
|
fetchMock.get("/api/v2/", "Restarted");
|
||
|
|
const { result, waitFor, waitForNextUpdate } = renderHook(() => useInstallPlugin(), {
|
||
|
|
wrapper: createWrapper(undefined, queryClient)
|
||
|
|
});
|
||
|
|
await act(() => {
|
||
|
|
const { install } = result.current;
|
||
|
|
install(availablePlugin, { restart: true, initialDelay: 5, timeout: 5 });
|
||
|
|
return waitForNextUpdate();
|
||
|
|
});
|
||
|
|
await waitFor(() => result.current.isInstalled);
|
||
|
|
expect(queryClient.getQueryState("plugins")!.isInvalidated).toBe(true);
|
||
|
|
});
|
||
|
|
|
||
|
|
it("should invalidate query keys", async () => {
|
||
|
|
const queryClient = createInfiniteCachingClient();
|
||
|
|
queryClient.setQueryData(["plugins", "available"], createPluginCollection([availablePlugin]));
|
||
|
|
queryClient.setQueryData(["plugins", "installed"], createPluginCollection([]));
|
||
|
|
queryClient.setQueryData(["plugins", "pending"], createPendingPlugins());
|
||
|
|
fetchMock.post("/api/v2/plugins/available/heart-of-gold-plugin/install", installedPlugin);
|
||
|
|
const { result, waitForNextUpdate } = renderHook(() => useInstallPlugin(), {
|
||
|
|
wrapper: createWrapper(undefined, queryClient)
|
||
|
|
});
|
||
|
|
await act(() => {
|
||
|
|
const { install } = result.current;
|
||
|
|
install(availablePlugin);
|
||
|
|
return waitForNextUpdate();
|
||
|
|
});
|
||
|
|
expect(queryClient.getQueryState("plugins")!.isInvalidated).toBe(true);
|
||
|
|
});
|
||
|
|
});
|
||
|
|
|
||
|
|
describe("useUninstallPlugin tests", () => {
|
||
|
|
it("should use restart parameter", async () => {
|
||
|
|
const queryClient = createInfiniteCachingClient();
|
||
|
|
queryClient.setQueryData(["plugins", "available"], createPluginCollection([]));
|
||
|
|
queryClient.setQueryData(["plugins", "installed"], createPluginCollection([installedPlugin]));
|
||
|
|
queryClient.setQueryData(["plugins", "pending"], createPendingPlugins());
|
||
|
|
fetchMock.post("/api/v2/plugins/installed/heart-of-gold-plugin/uninstall?restart=true", availablePlugin);
|
||
|
|
fetchMock.get("/api/v2/", "Restarted");
|
||
|
|
const { result, waitForNextUpdate, waitFor } = renderHook(() => useUninstallPlugin(), {
|
||
|
|
wrapper: createWrapper(undefined, queryClient)
|
||
|
|
});
|
||
|
|
await act(() => {
|
||
|
|
const { uninstall } = result.current;
|
||
|
|
uninstall(installedPlugin, { restart: true, initialDelay: 5, timeout: 5 });
|
||
|
|
return waitForNextUpdate();
|
||
|
|
});
|
||
|
|
await waitFor(() => result.current.isUninstalled);
|
||
|
|
expect(queryClient.getQueryState("plugins")!.isInvalidated).toBe(true);
|
||
|
|
});
|
||
|
|
|
||
|
|
it("should invalidate query keys", async () => {
|
||
|
|
const queryClient = createInfiniteCachingClient();
|
||
|
|
queryClient.setQueryData(["plugins", "available"], createPluginCollection([]));
|
||
|
|
queryClient.setQueryData(["plugins", "installed"], createPluginCollection([installedPlugin]));
|
||
|
|
queryClient.setQueryData(["plugins", "pending"], createPendingPlugins());
|
||
|
|
fetchMock.post("/api/v2/plugins/installed/heart-of-gold-plugin/uninstall", availablePlugin);
|
||
|
|
const { result, waitForNextUpdate } = renderHook(() => useUninstallPlugin(), {
|
||
|
|
wrapper: createWrapper(undefined, queryClient)
|
||
|
|
});
|
||
|
|
await act(() => {
|
||
|
|
const { uninstall } = result.current;
|
||
|
|
uninstall(installedPlugin);
|
||
|
|
return waitForNextUpdate();
|
||
|
|
});
|
||
|
|
expect(queryClient.getQueryState("plugins")!.isInvalidated).toBe(true);
|
||
|
|
});
|
||
|
|
});
|
||
|
|
|
||
|
|
describe("useUpdatePlugins tests", () => {
|
||
|
|
it("should use restart parameter", async () => {
|
||
|
|
const queryClient = createInfiniteCachingClient();
|
||
|
|
queryClient.setQueryData(["plugins", "available"], createPluginCollection([]));
|
||
|
|
queryClient.setQueryData(["plugins", "installed"], createPluginCollection([installedPlugin]));
|
||
|
|
queryClient.setQueryData(["plugins", "pending"], createPendingPlugins());
|
||
|
|
fetchMock.post("/api/v2/plugins/available/heart-of-gold-plugin/install?restart=true", installedPlugin);
|
||
|
|
fetchMock.get("/api/v2/", "Restarted");
|
||
|
|
const { result, waitForNextUpdate, waitFor } = renderHook(() => useUpdatePlugins(), {
|
||
|
|
wrapper: createWrapper(undefined, queryClient)
|
||
|
|
});
|
||
|
|
await act(() => {
|
||
|
|
const { update } = result.current;
|
||
|
|
update(installedPlugin, { restart: true, timeout: 5, initialDelay: 5 });
|
||
|
|
return waitForNextUpdate();
|
||
|
|
});
|
||
|
|
await waitFor(() => result.current.isUpdated);
|
||
|
|
expect(queryClient.getQueryState("plugins")!.isInvalidated).toBe(true);
|
||
|
|
});
|
||
|
|
it("should update collection", async () => {
|
||
|
|
const queryClient = createInfiniteCachingClient();
|
||
|
|
queryClient.setQueryData(["plugins", "available"], createPluginCollection([]));
|
||
|
|
queryClient.setQueryData(["plugins", "installed"], createPluginCollection([installedPlugin]));
|
||
|
|
queryClient.setQueryData(["plugins", "pending"], createPendingPlugins());
|
||
|
|
fetchMock.post("/api/v2/plugins/update", installedPlugin);
|
||
|
|
const { result, waitForNextUpdate } = renderHook(() => useUpdatePlugins(), {
|
||
|
|
wrapper: createWrapper(undefined, queryClient)
|
||
|
|
});
|
||
|
|
await act(() => {
|
||
|
|
const { update } = result.current;
|
||
|
|
update(createPluginCollection([installedPlugin, installedCorePlugin]));
|
||
|
|
return waitForNextUpdate();
|
||
|
|
});
|
||
|
|
expect(queryClient.getQueryState("plugins")!.isInvalidated).toBe(true);
|
||
|
|
});
|
||
|
|
it("should ignore restart parameter collection", async () => {
|
||
|
|
const queryClient = createInfiniteCachingClient();
|
||
|
|
queryClient.setQueryData(["plugins", "available"], createPluginCollection([]));
|
||
|
|
queryClient.setQueryData(["plugins", "installed"], createPluginCollection([installedPlugin]));
|
||
|
|
queryClient.setQueryData(["plugins", "pending"], createPendingPlugins());
|
||
|
|
fetchMock.post("/api/v2/plugins/update", installedPlugin);
|
||
|
|
const { result, waitForNextUpdate } = renderHook(() => useUpdatePlugins(), {
|
||
|
|
wrapper: createWrapper(undefined, queryClient)
|
||
|
|
});
|
||
|
|
await act(() => {
|
||
|
|
const { update } = result.current;
|
||
|
|
update(createPluginCollection([installedPlugin, installedCorePlugin]), { restart: true });
|
||
|
|
return waitForNextUpdate();
|
||
|
|
});
|
||
|
|
expect(queryClient.getQueryState("plugins")!.isInvalidated).toBe(true);
|
||
|
|
});
|
||
|
|
it("should invalidate query keys", async () => {
|
||
|
|
const queryClient = createInfiniteCachingClient();
|
||
|
|
queryClient.setQueryData(["plugins", "available"], createPluginCollection([]));
|
||
|
|
queryClient.setQueryData(["plugins", "installed"], createPluginCollection([installedPlugin]));
|
||
|
|
queryClient.setQueryData(["plugins", "pending"], createPendingPlugins());
|
||
|
|
fetchMock.post("/api/v2/plugins/available/heart-of-gold-plugin/install", installedPlugin);
|
||
|
|
const { result, waitForNextUpdate } = renderHook(() => useUpdatePlugins(), {
|
||
|
|
wrapper: createWrapper(undefined, queryClient)
|
||
|
|
});
|
||
|
|
await act(() => {
|
||
|
|
const { update } = result.current;
|
||
|
|
update(installedPlugin);
|
||
|
|
return waitForNextUpdate();
|
||
|
|
});
|
||
|
|
expect(queryClient.getQueryState("plugins")!.isInvalidated).toBe(true);
|
||
|
|
});
|
||
|
|
});
|
||
|
|
});
|