Files
SCM-Manager/scm-ui/ui-extensions/src/binder.test.ts

201 lines
7.9 KiB
TypeScript
Raw Normal View History

/*
* 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 { Binder, ExtensionPointDefinition, SimpleDynamicExtensionPointDefinition } from "./binder";
2019-10-19 17:15:53 +02:00
describe("binder tests", () => {
let binder: Binder;
beforeEach(() => {
binder = new Binder("testing");
});
2019-10-19 17:15:53 +02:00
it("should return an empty array for non existing extension points", () => {
const extensions = binder.getExtensions("hitchhiker");
expect(extensions).toEqual([]);
});
2019-10-19 17:15:53 +02:00
it("should return the binded extensions", () => {
binder.bind("hitchhiker.trillian", "heartOfGold");
binder.bind("hitchhiker.trillian", "earth");
const extensions = binder.getExtensions("hitchhiker.trillian");
2019-10-19 17:15:53 +02:00
expect(extensions).toEqual(["heartOfGold", "earth"]);
});
2019-10-19 17:15:53 +02:00
it("should return the first bound extension", () => {
binder.bind("hitchhiker.trillian", "heartOfGold");
binder.bind("hitchhiker.trillian", "earth");
expect(binder.getExtension("hitchhiker.trillian")).toBe("heartOfGold");
});
2019-10-19 17:15:53 +02:00
it("should return null if no extension was bound", () => {
expect(binder.getExtension("hitchhiker.trillian")).toBe(null);
});
2019-10-19 17:15:53 +02:00
it("should return true, if an extension is bound", () => {
binder.bind("hitchhiker.trillian", "heartOfGold");
expect(binder.hasExtension("hitchhiker.trillian")).toBe(true);
});
2019-10-19 17:15:53 +02:00
it("should return false, if no extension is bound", () => {
expect(binder.hasExtension("hitchhiker.trillian")).toBe(false);
});
2019-10-19 17:15:53 +02:00
type Props = {
2019-10-21 10:57:56 +02:00
category: string;
2019-10-19 17:15:53 +02:00
};
it("should return only extensions which predicates matches", () => {
binder.bind("hitchhiker.trillian", "heartOfGold", (props: Props) => props.category === "a");
binder.bind("hitchhiker.trillian", "earth", (props: Props) => props.category === "b");
binder.bind("hitchhiker.trillian", "earth2", (props: Props) => props.category === "a");
const extensions = binder.getExtensions("hitchhiker.trillian", {
category: "b",
});
2019-10-19 17:15:53 +02:00
expect(extensions).toEqual(["earth"]);
});
it("should return extensions in ascending order", () => {
binder.bind("hitchhiker.trillian", "planetA", () => true, "zeroWaste");
binder.bind("hitchhiker.trillian", "planetB", () => true, "EPSILON");
binder.bind("hitchhiker.trillian", "planetC", () => true, "emptyBin");
binder.bind("hitchhiker.trillian", "planetD", () => true, "absolute");
const extensions = binder.getExtensions("hitchhiker.trillian");
expect(extensions).toEqual(["planetD", "planetC", "planetB", "planetA"]);
});
it("should return extensions starting with entries with specified extensionName", () => {
binder.bind("hitchhiker.trillian", "planetA", () => true);
binder.bind("hitchhiker.trillian", "planetB", () => true, "zeroWaste");
binder.bind("hitchhiker.trillian", "planetC", () => true);
binder.bind("hitchhiker.trillian", "planetD", () => true, "emptyBin");
const extensions = binder.getExtensions("hitchhiker.trillian");
expect(extensions[0]).toEqual("planetD");
expect(extensions[1]).toEqual("planetB");
});
it("should allow typings for extension points but still be backwards-compatible", () => {
type TestExtensionPointA = ExtensionPointDefinition<"test.extension.a", number, undefined>;
type TestExtensionPointB = ExtensionPointDefinition<"test.extension.b", number, { testProp: boolean[] }>;
binder.bind<TestExtensionPointA>("test.extension.a", 2, () => false);
const binderExtensionA = binder.getExtension<TestExtensionPointA>("test.extension.a");
expect(binderExtensionA).not.toBeNull();
binder.bind<TestExtensionPointB>("test.extension.b", 2);
const binderExtensionsB = binder.getExtensions<TestExtensionPointB>("test.extension.b", {
testProp: [true, false],
});
expect(binderExtensionsB).toHaveLength(1);
binder.bind("test.extension.c", 2, () => false);
const binderExtensionC = binder.getExtension("test.extension.c");
expect(binderExtensionC).not.toBeNull();
});
it("should allow typings for dynamic extension points", () => {
type MarkdownCodeLanguageRendererProps = {
language?: string;
value: string;
};
type MarkdownCodeLanguageRendererExtensionPoint<S extends string | undefined = undefined> =
SimpleDynamicExtensionPointDefinition<
"markdown-renderer.code.",
(props: any) => any,
MarkdownCodeLanguageRendererProps,
S
>;
type UmlExtensionPoint = MarkdownCodeLanguageRendererExtensionPoint<"uml">;
binder.bind<UmlExtensionPoint>("markdown-renderer.code.uml", (props) => props.value);
const language = "uml";
const extensionPointName = `markdown-renderer.code.${language}` as const;
const dynamicExtension = binder.getExtension<MarkdownCodeLanguageRendererExtensionPoint>(extensionPointName, {
language: "uml",
value: "const a = 2;",
});
expect(dynamicExtension).not.toBeNull();
});
it("should allow options parameter", () => {
binder.bind("hitchhiker.trillian", "planetA", {
predicate: () => true,
extensionName: "zeroWaste",
});
const extensions = binder.getExtensions("hitchhiker.trillian");
expect(extensions).toEqual(["planetA"]);
});
it("should allow empty options parameter", () => {
binder.bind("hitchhiker.trillian", "planetA", {});
const extensions = binder.getExtensions("hitchhiker.trillian");
expect(extensions).toEqual(["planetA"]);
});
it("should allow options parameter with only predicate", () => {
binder.bind("hitchhiker.trillian", "planetA", {
predicate: () => true,
});
const extensions = binder.getExtensions("hitchhiker.trillian");
expect(extensions).toEqual(["planetA"]);
});
it("should allow options parameter with only extensionName", () => {
binder.bind("hitchhiker.trillian", "planetA", {
extensionName: "zeroWaste",
});
const extensions = binder.getExtensions("hitchhiker.trillian");
expect(extensions).toEqual(["planetA"]);
});
it("should order by priority in descending order", () => {
binder.bind("hitchhiker.trillian", "planetA", { priority: 10 });
binder.bind("hitchhiker.trillian", "planetB", { priority: 50 });
binder.bind("hitchhiker.trillian", "planetC", { priority: 100 });
binder.bind("hitchhiker.trillian", "planetD", { priority: 75 });
const extensions = binder.getExtensions("hitchhiker.trillian");
expect(extensions).toEqual(["planetC", "planetD", "planetB", "planetA"]);
});
it("should order by priority over ordering by name", () => {
binder.bind("hitchhiker.trillian", "planetA", { priority: 10, extensionName: "ignore" });
binder.bind("hitchhiker.trillian", "planetB", { priority: 50 });
binder.bind("hitchhiker.trillian", "planetC", { priority: 100, extensionName: "me" });
binder.bind("hitchhiker.trillian", "planetD", { priority: 75 });
const extensions = binder.getExtensions("hitchhiker.trillian");
expect(extensions).toEqual(["planetC", "planetD", "planetB", "planetA"]);
});
});