mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
- Upgrade to jest 28 - Switch to using swc - Use @side/jest-runtime as the jest-runtime Signed-off-by: Sebastian Malton <sebastian@malton.name>
719 lines
19 KiB
TypeScript
719 lines
19 KiB
TypeScript
/**
|
|
* Copyright (c) OpenLens Authors. All rights reserved.
|
|
* Licensed under MIT License. See LICENSE in root directory for more information.
|
|
*/
|
|
|
|
import { forRemoteCluster, KubeApi } from "../kube-api";
|
|
import { KubeJsonApi } from "../kube-json-api";
|
|
import { KubeObject } from "../kube-object";
|
|
import AbortController from "abort-controller";
|
|
import { delay } from "../../utils/delay";
|
|
import { PassThrough } from "stream";
|
|
import { ApiManager } from "../api-manager";
|
|
import type { FetchMock } from "jest-fetch-mock/types";
|
|
import { DeploymentApi, Ingress, IngressApi, Pod, PodApi } from "../endpoints";
|
|
import { getDiForUnitTesting } from "../../../renderer/getDiForUnitTesting";
|
|
import apiManagerInjectable from "../api-manager/manager.injectable";
|
|
import autoRegistrationInjectable from "../api-manager/auto-registration.injectable";
|
|
|
|
jest.mock("../api-manager");
|
|
|
|
const mockFetch = fetch as FetchMock;
|
|
|
|
describe("forRemoteCluster", () => {
|
|
let apiManager: jest.Mocked<ApiManager>;
|
|
|
|
beforeEach(() => {
|
|
const di = getDiForUnitTesting({ doGeneralOverrides: true });
|
|
|
|
apiManager = new ApiManager() as jest.Mocked<ApiManager>;
|
|
|
|
di.override(apiManagerInjectable, () => apiManager);
|
|
});
|
|
|
|
it("builds api client for KubeObject", async () => {
|
|
const api = forRemoteCluster({
|
|
cluster: {
|
|
server: "https://127.0.0.1:6443",
|
|
},
|
|
user: {
|
|
token: "daa",
|
|
},
|
|
}, Pod);
|
|
|
|
expect(api).toBeInstanceOf(KubeApi);
|
|
});
|
|
|
|
it("builds api client for given KubeApi", async () => {
|
|
const api = forRemoteCluster({
|
|
cluster: {
|
|
server: "https://127.0.0.1:6443",
|
|
},
|
|
user: {
|
|
token: "daa",
|
|
},
|
|
}, Pod, PodApi);
|
|
|
|
expect(api).toBeInstanceOf(PodApi);
|
|
});
|
|
|
|
it("calls right api endpoint", async () => {
|
|
const api = forRemoteCluster({
|
|
cluster: {
|
|
server: "https://127.0.0.1:6443",
|
|
},
|
|
user: {
|
|
token: "daa",
|
|
},
|
|
}, Pod);
|
|
|
|
mockFetch.mockResponse(async (request: any) => {
|
|
expect(request.url).toEqual("https://127.0.0.1:6443/api/v1/pods");
|
|
|
|
return {
|
|
body: "hello",
|
|
};
|
|
});
|
|
|
|
expect.hasAssertions();
|
|
|
|
await api.list();
|
|
});
|
|
});
|
|
|
|
describe("KubeApi", () => {
|
|
let request: KubeJsonApi;
|
|
let apiManager: jest.Mocked<ApiManager>;
|
|
|
|
beforeEach(() => {
|
|
const di = getDiForUnitTesting({ doGeneralOverrides: true });
|
|
|
|
request = new KubeJsonApi({
|
|
serverAddress: `http://127.0.0.1:9999`,
|
|
apiBase: "/api-kube",
|
|
});
|
|
apiManager = new ApiManager() as jest.Mocked<ApiManager>;
|
|
|
|
di.override(apiManagerInjectable, () => apiManager);
|
|
di.inject(autoRegistrationInjectable);
|
|
});
|
|
|
|
it("uses url from apiBase if apiBase contains the resource", async () => {
|
|
mockFetch.mockResponse(async (request: any) => {
|
|
if (request.url === "http://127.0.0.1:9999/api-kube/apis/networking.k8s.io/v1") {
|
|
return {
|
|
body: JSON.stringify({
|
|
resources: [{
|
|
name: "ingresses",
|
|
}],
|
|
}),
|
|
};
|
|
} else if (request.url === "http://127.0.0.1:9999/api-kube/apis/extensions/v1beta1") {
|
|
// Even if the old API contains ingresses, KubeApi should prefer the apiBase url
|
|
return {
|
|
body: JSON.stringify({
|
|
resources: [{
|
|
name: "ingresses",
|
|
}],
|
|
}),
|
|
};
|
|
} else {
|
|
return {
|
|
body: JSON.stringify({
|
|
resources: [],
|
|
}),
|
|
};
|
|
}
|
|
});
|
|
|
|
const apiBase = "/apis/networking.k8s.io/v1/ingresses";
|
|
const fallbackApiBase = "/apis/extensions/v1beta1/ingresses";
|
|
const kubeApi = new IngressApi({
|
|
request,
|
|
objectConstructor: Ingress,
|
|
apiBase,
|
|
fallbackApiBases: [fallbackApiBase],
|
|
checkPreferredVersion: true,
|
|
});
|
|
|
|
await kubeApi.get({
|
|
name: "foo",
|
|
namespace: "default",
|
|
});
|
|
expect(kubeApi.apiPrefix).toEqual("/apis");
|
|
expect(kubeApi.apiGroup).toEqual("networking.k8s.io");
|
|
});
|
|
|
|
it("uses url from fallbackApiBases if apiBase lacks the resource", async () => {
|
|
mockFetch.mockResponse(async (request: any) => {
|
|
if (request.url === "http://127.0.0.1:9999/api-kube/apis/networking.k8s.io/v1") {
|
|
return {
|
|
body: JSON.stringify({
|
|
resources: [],
|
|
}),
|
|
};
|
|
} else if (request.url === "http://127.0.0.1:9999/api-kube/apis/extensions/v1beta1") {
|
|
return {
|
|
body: JSON.stringify({
|
|
resources: [{
|
|
name: "ingresses",
|
|
}],
|
|
}),
|
|
};
|
|
} else {
|
|
return {
|
|
body: JSON.stringify({
|
|
resources: [],
|
|
}),
|
|
};
|
|
}
|
|
});
|
|
|
|
const apiBase = "apis/networking.k8s.io/v1/ingresses";
|
|
const fallbackApiBase = "/apis/extensions/v1beta1/ingresses";
|
|
const kubeApi = new IngressApi({
|
|
request,
|
|
objectConstructor: Object.assign(KubeObject, { apiBase }),
|
|
kind: "Ingress",
|
|
fallbackApiBases: [fallbackApiBase],
|
|
checkPreferredVersion: true,
|
|
});
|
|
|
|
await kubeApi.get({
|
|
name: "foo",
|
|
namespace: "default",
|
|
});
|
|
expect(kubeApi.apiPrefix).toEqual("/apis");
|
|
expect(kubeApi.apiGroup).toEqual("extensions");
|
|
});
|
|
|
|
describe("checkPreferredVersion", () => {
|
|
it("registers with apiManager if checkPreferredVersion changes apiVersionPreferred", async () => {
|
|
expect.hasAssertions();
|
|
|
|
const api = new IngressApi({
|
|
objectConstructor: Ingress,
|
|
checkPreferredVersion: true,
|
|
fallbackApiBases: ["/apis/extensions/v1beta1/ingresses"],
|
|
request: {
|
|
get: jest.fn()
|
|
.mockImplementation((path: string) => {
|
|
switch (path) {
|
|
case "/apis/networking.k8s.io/v1":
|
|
throw new Error("no");
|
|
case "/apis/extensions/v1beta1":
|
|
return {
|
|
resources: [
|
|
{
|
|
name: "ingresses",
|
|
},
|
|
],
|
|
};
|
|
case "/apis/extensions":
|
|
return {
|
|
preferredVersion: {
|
|
version: "v1beta1",
|
|
},
|
|
};
|
|
default:
|
|
throw new Error("unknown path");
|
|
}
|
|
}),
|
|
} as Partial<KubeJsonApi> as KubeJsonApi,
|
|
});
|
|
|
|
await (api as any).checkPreferredVersion();
|
|
|
|
expect(api.apiVersionPreferred).toBe("v1beta1");
|
|
expect(apiManager.registerApi).toBeCalledWith(api);
|
|
});
|
|
|
|
it("registers with apiManager if checkPreferredVersion changes apiVersionPreferred with non-grouped apis", async () => {
|
|
expect.hasAssertions();
|
|
|
|
const api = new PodApi({
|
|
objectConstructor: Pod,
|
|
checkPreferredVersion: true,
|
|
fallbackApiBases: ["/api/v1beta1/pods"],
|
|
request: {
|
|
get: jest.fn()
|
|
.mockImplementation((path: string) => {
|
|
switch (path) {
|
|
case "/api/v1":
|
|
throw new Error("no");
|
|
case "/api/v1beta1":
|
|
return {
|
|
resources: [
|
|
{
|
|
name: "pods",
|
|
},
|
|
],
|
|
};
|
|
case "/api":
|
|
return {
|
|
preferredVersion: {
|
|
version: "v1beta1",
|
|
},
|
|
};
|
|
default:
|
|
throw new Error("unknown path");
|
|
}
|
|
}),
|
|
} as Partial<KubeJsonApi> as KubeJsonApi,
|
|
});
|
|
|
|
await (api as any).checkPreferredVersion();
|
|
|
|
expect(api.apiVersionPreferred).toBe("v1beta1");
|
|
expect(apiManager.registerApi).toBeCalledWith(api);
|
|
});
|
|
});
|
|
|
|
describe("patch", () => {
|
|
let api: DeploymentApi;
|
|
|
|
beforeEach(() => {
|
|
api = new DeploymentApi({
|
|
request,
|
|
});
|
|
});
|
|
|
|
it("sends strategic patch by default", async () => {
|
|
expect.hasAssertions();
|
|
|
|
mockFetch.mockResponse(async request => {
|
|
expect(request.method).toEqual("PATCH");
|
|
expect(request.headers.get("content-type")).toMatch("strategic-merge-patch");
|
|
expect(request.body?.toString()).toEqual(JSON.stringify({ spec: { replicas: 2 }}));
|
|
|
|
return {};
|
|
});
|
|
|
|
await api.patch({ name: "test", namespace: "default" }, {
|
|
spec: { replicas: 2 },
|
|
});
|
|
});
|
|
|
|
it("allows to use merge patch", async () => {
|
|
expect.hasAssertions();
|
|
|
|
mockFetch.mockResponse(async request => {
|
|
expect(request.method).toEqual("PATCH");
|
|
expect(request.headers.get("content-type")).toMatch("merge-patch");
|
|
expect(request.body?.toString()).toEqual(JSON.stringify({ spec: { replicas: 2 }}));
|
|
|
|
return {};
|
|
});
|
|
|
|
await api.patch({ name: "test", namespace: "default" }, {
|
|
spec: { replicas: 2 },
|
|
}, "merge");
|
|
});
|
|
|
|
it("allows to use json patch", async () => {
|
|
expect.hasAssertions();
|
|
|
|
mockFetch.mockResponse(async request => {
|
|
expect(request.method).toEqual("PATCH");
|
|
expect(request.headers.get("content-type")).toMatch("json-patch");
|
|
expect(request.body?.toString()).toEqual(JSON.stringify([{ op: "replace", path: "/spec/replicas", value: 2 }]));
|
|
|
|
return {};
|
|
});
|
|
|
|
await api.patch({ name: "test", namespace: "default" }, [
|
|
{ op: "replace", path: "/spec/replicas", value: 2 },
|
|
], "json");
|
|
});
|
|
|
|
it("allows deep partial patch", async () => {
|
|
expect.hasAssertions();
|
|
|
|
mockFetch.mockResponse(async request => {
|
|
expect(request.method).toEqual("PATCH");
|
|
expect(request.headers.get("content-type")).toMatch("merge-patch");
|
|
expect(request.body?.toString()).toEqual(JSON.stringify({ metadata: { annotations: { provisioned: "true" }}}));
|
|
|
|
return {};
|
|
});
|
|
|
|
await api.patch(
|
|
{ name: "test", namespace: "default" },
|
|
{ metadata: { annotations: { provisioned: "true" }}},
|
|
"merge",
|
|
);
|
|
});
|
|
});
|
|
|
|
describe("delete", () => {
|
|
let api: PodApi;
|
|
|
|
beforeEach(() => {
|
|
api = new PodApi({
|
|
request,
|
|
objectConstructor: Pod,
|
|
});
|
|
});
|
|
|
|
it("sends correct request with empty namespace", async () => {
|
|
expect.hasAssertions();
|
|
mockFetch.mockResponse(async request => {
|
|
expect(request.method).toEqual("DELETE");
|
|
expect(request.url).toEqual("http://127.0.0.1:9999/api-kube/api/v1/pods/foo?propagationPolicy=Background");
|
|
|
|
return {};
|
|
});
|
|
|
|
await api.delete({ name: "foo", namespace: "" });
|
|
});
|
|
|
|
it("sends correct request without namespace", async () => {
|
|
expect.hasAssertions();
|
|
mockFetch.mockResponse(async request => {
|
|
expect(request.method).toEqual("DELETE");
|
|
expect(request.url).toEqual("http://127.0.0.1:9999/api-kube/api/v1/namespaces/default/pods/foo?propagationPolicy=Background");
|
|
|
|
return {};
|
|
});
|
|
|
|
await api.delete({ name: "foo" });
|
|
});
|
|
|
|
it("sends correct request with namespace", async () => {
|
|
expect.hasAssertions();
|
|
mockFetch.mockResponse(async request => {
|
|
expect(request.method).toEqual("DELETE");
|
|
expect(request.url).toEqual("http://127.0.0.1:9999/api-kube/api/v1/namespaces/kube-system/pods/foo?propagationPolicy=Background");
|
|
|
|
return {};
|
|
});
|
|
|
|
await api.delete({ name: "foo", namespace: "kube-system" });
|
|
});
|
|
|
|
it("allows to change propagationPolicy", async () => {
|
|
expect.hasAssertions();
|
|
mockFetch.mockResponse(async request => {
|
|
expect(request.method).toEqual("DELETE");
|
|
expect(request.url).toMatch("propagationPolicy=Orphan");
|
|
|
|
return {};
|
|
});
|
|
|
|
await api.delete({ name: "foo", namespace: "default", propagationPolicy: "Orphan" });
|
|
});
|
|
});
|
|
|
|
describe("watch", () => {
|
|
let api: PodApi;
|
|
let stream: PassThrough;
|
|
|
|
beforeEach(() => {
|
|
api = new PodApi({
|
|
request,
|
|
objectConstructor: Pod,
|
|
});
|
|
stream = new PassThrough();
|
|
});
|
|
|
|
afterEach(() => {
|
|
stream.end();
|
|
stream.destroy();
|
|
});
|
|
|
|
it("sends a valid watch request", () => {
|
|
const spy = jest.spyOn(request, "getResponse");
|
|
|
|
mockFetch.mockResponse(async () => {
|
|
return {
|
|
// needed for https://github.com/jefflau/jest-fetch-mock/issues/218
|
|
body: stream as unknown as string,
|
|
};
|
|
});
|
|
|
|
api.watch({ namespace: "kube-system" });
|
|
expect(spy).toHaveBeenCalledWith("/api/v1/namespaces/kube-system/pods?watch=1&resourceVersion=", expect.anything(), expect.anything());
|
|
});
|
|
|
|
it("sends timeout as a query parameter", async () => {
|
|
const spy = jest.spyOn(request, "getResponse");
|
|
|
|
mockFetch.mockResponse(async () => {
|
|
return {
|
|
// needed for https://github.com/jefflau/jest-fetch-mock/issues/218
|
|
body: stream as unknown as string,
|
|
};
|
|
});
|
|
|
|
api.watch({ namespace: "kube-system", timeout: 60 });
|
|
expect(spy).toHaveBeenCalledWith("/api/v1/namespaces/kube-system/pods?watch=1&resourceVersion=", { query: { timeoutSeconds: 60 }}, expect.anything());
|
|
});
|
|
|
|
it("aborts watch using abortController", (done) => {
|
|
const spy = jest.spyOn(request, "getResponse");
|
|
|
|
mockFetch.mockResponse(async request => {
|
|
request.signal.addEventListener("abort", () => {
|
|
done();
|
|
});
|
|
|
|
return {
|
|
// needed for https://github.com/jefflau/jest-fetch-mock/issues/218
|
|
body: stream as unknown as string,
|
|
};
|
|
});
|
|
|
|
const abortController = new AbortController();
|
|
|
|
api.watch({
|
|
namespace: "kube-system",
|
|
timeout: 60,
|
|
abortController,
|
|
});
|
|
|
|
expect(spy).toHaveBeenCalledWith("/api/v1/namespaces/kube-system/pods?watch=1&resourceVersion=", { query: { timeoutSeconds: 60 }}, expect.anything());
|
|
delay(100).then(() => abortController.abort());
|
|
});
|
|
|
|
describe("retries", () => {
|
|
it("if request ended", (done) => {
|
|
const spy = jest.spyOn(request, "getResponse");
|
|
|
|
jest.spyOn(stream, "on").mockImplementation((event: string | symbol, callback: Function) => {
|
|
// End the request in 100ms.
|
|
if (event === "end") {
|
|
setTimeout(() => {
|
|
callback();
|
|
}, 100);
|
|
}
|
|
|
|
return stream;
|
|
});
|
|
|
|
// we need to mock using jest as jest-fetch-mock doesn't support mocking the body completely
|
|
jest.spyOn(global, "fetch").mockImplementation(async () => {
|
|
return {
|
|
ok: true,
|
|
body: stream as never,
|
|
} as Partial<Response> as Response;
|
|
});
|
|
|
|
api.watch({
|
|
namespace: "kube-system",
|
|
});
|
|
|
|
expect(spy).toHaveBeenCalledTimes(1);
|
|
|
|
setTimeout(() => {
|
|
expect(spy).toHaveBeenCalledTimes(2);
|
|
done();
|
|
}, 2000);
|
|
});
|
|
|
|
it("if request not closed after timeout", (done) => {
|
|
const spy = jest.spyOn(request, "getResponse");
|
|
|
|
mockFetch.mockResponse(async () => {
|
|
return {
|
|
// needed for https://github.com/jefflau/jest-fetch-mock/issues/218
|
|
body: stream as unknown as string,
|
|
};
|
|
});
|
|
|
|
const timeoutSeconds = 1;
|
|
|
|
api.watch({
|
|
namespace: "kube-system",
|
|
timeout: timeoutSeconds,
|
|
});
|
|
|
|
expect(spy).toHaveBeenCalledTimes(1);
|
|
|
|
setTimeout(() => {
|
|
expect(spy).toHaveBeenCalledTimes(2);
|
|
done();
|
|
}, timeoutSeconds * 1000 * 1.2);
|
|
});
|
|
|
|
it("retries only once if request ends and timeout is set", (done) => {
|
|
const spy = jest.spyOn(request, "getResponse");
|
|
|
|
jest.spyOn(stream, "on").mockImplementation((event: string | symbol, callback: Function) => {
|
|
// End the request in 100ms.
|
|
if (event === "end") {
|
|
setTimeout(() => {
|
|
callback();
|
|
}, 100);
|
|
}
|
|
|
|
return stream;
|
|
});
|
|
|
|
// we need to mock using jest as jest-fetch-mock doesn't support mocking the body completely
|
|
jest.spyOn(global, "fetch").mockImplementation(async () => {
|
|
return {
|
|
ok: true,
|
|
body: stream as never,
|
|
} as Partial<Response> as Response;
|
|
});
|
|
|
|
const timeoutSeconds = 0.5;
|
|
|
|
api.watch({
|
|
namespace: "kube-system",
|
|
timeout: timeoutSeconds,
|
|
});
|
|
|
|
expect(spy).toHaveBeenCalledTimes(1);
|
|
|
|
setTimeout(() => {
|
|
expect(spy).toHaveBeenCalledTimes(2);
|
|
done();
|
|
}, 2000);
|
|
});
|
|
|
|
afterEach(() => {
|
|
jest.clearAllMocks();
|
|
});
|
|
});
|
|
});
|
|
|
|
describe("create", () => {
|
|
let api: PodApi;
|
|
|
|
beforeEach(() => {
|
|
api = new PodApi({
|
|
request,
|
|
objectConstructor: Pod,
|
|
});
|
|
});
|
|
|
|
it("should add kind and apiVersion", async () => {
|
|
expect.hasAssertions();
|
|
|
|
mockFetch.mockResponse(async request => {
|
|
expect(request.method).toEqual("POST");
|
|
expect(JSON.parse(String(request.body))).toEqual({
|
|
kind: "Pod",
|
|
apiVersion: "v1",
|
|
metadata: {
|
|
name: "foobar",
|
|
namespace: "default",
|
|
},
|
|
spec: {
|
|
containers: [
|
|
{
|
|
name: "web",
|
|
image: "nginx",
|
|
ports: [
|
|
{
|
|
name: "web",
|
|
containerPort: 80,
|
|
protocol: "TCP",
|
|
},
|
|
],
|
|
},
|
|
],
|
|
},
|
|
});
|
|
|
|
return {};
|
|
});
|
|
|
|
await api.create({
|
|
name: "foobar",
|
|
namespace: "default",
|
|
}, {
|
|
spec: {
|
|
containers: [
|
|
{
|
|
name: "web",
|
|
image: "nginx",
|
|
ports: [
|
|
{
|
|
name: "web",
|
|
containerPort: 80,
|
|
protocol: "TCP",
|
|
},
|
|
],
|
|
},
|
|
],
|
|
},
|
|
});
|
|
});
|
|
|
|
it("doesn't override metadata.labels", async () => {
|
|
expect.hasAssertions();
|
|
|
|
mockFetch.mockResponse(async request => {
|
|
expect(request.method).toEqual("POST");
|
|
expect(JSON.parse(String(request.body))).toEqual({
|
|
kind: "Pod",
|
|
apiVersion: "v1",
|
|
metadata: {
|
|
name: "foobar",
|
|
namespace: "default",
|
|
labels: {
|
|
foo: "bar",
|
|
},
|
|
},
|
|
});
|
|
|
|
return {};
|
|
});
|
|
|
|
await api.create({
|
|
name: "foobar",
|
|
namespace: "default",
|
|
}, {
|
|
metadata: {
|
|
labels: {
|
|
foo: "bar",
|
|
},
|
|
},
|
|
});
|
|
});
|
|
});
|
|
|
|
describe("update", () => {
|
|
let api: PodApi;
|
|
|
|
beforeEach(() => {
|
|
api = new PodApi({
|
|
request,
|
|
objectConstructor: Pod,
|
|
});
|
|
});
|
|
|
|
it("doesn't override metadata.labels", async () => {
|
|
expect.hasAssertions();
|
|
|
|
mockFetch.mockResponse(async request => {
|
|
expect(request.method).toEqual("PUT");
|
|
expect(JSON.parse(String(request.body))).toEqual({
|
|
metadata: {
|
|
name: "foobar",
|
|
namespace: "default",
|
|
labels: {
|
|
foo: "bar",
|
|
},
|
|
},
|
|
});
|
|
|
|
return {};
|
|
});
|
|
|
|
await api.update({
|
|
name: "foobar",
|
|
namespace: "default",
|
|
}, {
|
|
metadata: {
|
|
labels: {
|
|
foo: "bar",
|
|
},
|
|
},
|
|
});
|
|
});
|
|
});
|
|
});
|