Add address prefix testing functions (#352)

Summary:
These functions can be used for address filtering: finding all nodes
owned by a plugin, or all edges of a certain plugin-type, or similar.
Clients can implement this in terms of `toParts`, but when the
underlying type of the opaque type is known there exists a simpler and
more efficient implementation.

Test Plan:
Unit tests added. Run `yarn travis`.

wchargin-branch: address-prefix
This commit is contained in:
William Chargin 2018-06-06 11:49:54 -07:00 committed by GitHub
parent bc98383053
commit 4a06485a99
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 138 additions and 0 deletions

View File

@ -142,3 +142,31 @@ export function edgeToString(a: EdgeAddress): string {
const parts = toParts(a);
return `edgeAddress(${stringify(parts)})`;
}
/**
* Determine whether the parts of `prefix` form a prefix of the parts of
* `address`. That is, determine whether there exists an `i` such that
* `toParts(prefix)` equals `toParts(address).slice(0, i)`.
*/
export function nodeHasPrefix(
address: NodeAddress,
prefix: NodeAddress
): boolean {
assertNodeAddress(address);
assertNodeAddress(prefix);
return address.startsWith(prefix);
}
/**
* Determine whether the parts of `prefix` form a prefix of the parts of
* `address`. That is, determine whether there exists an `i` such that
* `toParts(prefix)` equals `toParts(address).slice(0, i)`.
*/
export function edgeHasPrefix(
address: EdgeAddress,
prefix: EdgeAddress
): boolean {
assertEdgeAddress(address);
assertEdgeAddress(prefix);
return address.startsWith(prefix);
}

View File

@ -6,9 +6,11 @@ import {
assertEdgeAddress,
edgeAddress,
edgeAppend,
edgeHasPrefix,
edgeToString,
nodeAddress,
nodeAppend,
nodeHasPrefix,
nodeToString,
toParts,
} from "./_address";
@ -188,6 +190,114 @@ describe("core/address", () => {
checkToString(nodeToString, "NodeAddress", nodeAddress, edgeAddress);
checkToString(edgeToString, "EdgeAddress", edgeAddress, nodeAddress);
function checkHasPrefix<
Good: NodeAddress | EdgeAddress,
Bad: NodeAddress | EdgeAddress
>(
hasPrefix: (Good, Good) => boolean,
kind: "NodeAddress" | "EdgeAddress",
goodConstructor: (string[]) => Good,
badConstructor: (string[]) => Bad
) {
describe(hasPrefix.name, () => {
describe("errors on", () => {
[null, undefined].forEach((bad) => {
it(`${String(bad)} base input`, () => {
// $ExpectFlowError
expect(() => hasPrefix(bad)).toThrow(String(bad));
});
});
it("wrong kind", () => {
// $ExpectFlowError
expect(() => hasPrefix(badConstructor(["foo"]))).toThrow(
`expected ${kind}`
);
});
});
const address = goodConstructor;
it("accepts the empty prefix of non-empty input", () => {
expect(hasPrefix(address(["foo", "bar"]), address([]))).toBe(true);
});
it("accepts the empty prefix of empty input", () => {
expect(hasPrefix(address([]), address([]))).toBe(true);
});
it("rejects a non-empty prefix of empty input", () => {
expect(hasPrefix(address([]), address(["foo", "bar"]))).toBe(false);
});
it("accepts a normal input", () => {
expect(
hasPrefix(address(["foo", "bar", "baz"]), address(["foo", "bar"]))
).toBe(true);
});
it("accepts that an address is a prefix of itself", () => {
expect(
hasPrefix(address(["foo", "bar"]), address(["foo", "bar"]))
).toBe(true);
});
it("accepts inputs with empty components", () => {
expect(
hasPrefix(
address(["foo", "", "bar", "", "baz"]),
address(["foo", "", "bar", ""])
)
).toBe(true);
});
it("rejects inputs with no nontrivial common prefix", () => {
expect(
hasPrefix(address(["foo", "bar", "baz"]), address(["bar", "foo"]))
).toBe(false);
});
it("rejects inputs with insufficiently long common prefix", () => {
expect(
hasPrefix(address(["foo", "bar", "baz"]), address(["foo", "quux"]))
).toBe(false);
});
it("rejects when the putative prefix is a proper infix", () => {
expect(
hasPrefix(address(["foo", "bar", "baz"]), address(["bar"]))
).toBe(false);
});
it("rejects when the putative prefix is a proper suffix", () => {
expect(
hasPrefix(address(["foo", "bar", "baz"]), address(["bar", "baz"]))
).toBe(false);
});
it("rejects when the arguments are reversed", () => {
expect(
hasPrefix(address(["foo", "bar"]), address(["foo", "bar", "baz"]))
).toBe(false);
});
it("rejects when the last component is truncated", () => {
expect(
hasPrefix(address(["foo", "bar", "baz"]), address(["foo", "ba"]))
).toBe(false);
});
it("rejects when two components have been concatenated", () => {
expect(
hasPrefix(address(["foo", "bar", "baz"]), address(["foobar", "baz"]))
).toBe(false);
});
it("rejects an extra empty component in the middle of the base", () => {
expect(
hasPrefix(address(["foo", "", "baz"]), address(["foo", "baz"]))
).toBe(false);
});
it("rejects an extra empty component in the middle of the prefix", () => {
expect(
hasPrefix(address(["foo", "baz"]), address(["foo", "", "baz"]))
).toBe(false);
});
it("rejects an extra empty component at the end of the prefix", () => {
expect(
hasPrefix(address(["foo", "baz"]), address(["foo", "baz", ""]))
).toBe(false);
});
});
}
checkHasPrefix(nodeHasPrefix, "NodeAddress", nodeAddress, edgeAddress);
checkHasPrefix(edgeHasPrefix, "EdgeAddress", edgeAddress, nodeAddress);
describe("type assertions", () => {
function checkAssertion(f, good, bad, badMsg) {
describe(f.name, () => {