mirror of
https://github.com/status-im/sourcecred.git
synced 2025-01-14 14:46:30 +00:00
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:
parent
bc98383053
commit
4a06485a99
@ -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);
|
||||
}
|
||||
|
@ -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, () => {
|
||||
|
Loading…
x
Reference in New Issue
Block a user