Weights: Refactor around a new EdgeWeight type (#1144)

This refactors the weights module (and downstream consumers) so that
rather than tracking forwardWeight and backwardsWeight as separate
fields, we have an `EdgeWeight` type with a forwards and backwards
property. I feel this makes the code a little cleaner and wanted it a
few times while doing other refactors in this module.

Note that the graphToMarkovChain module has a distinct `EdgeWeight` type
which serves a similar purpose but happens to have different field
names. I've added a TODO to rename those fields for consistency.

Test plan: Since this is slight data structure re-organization, no new
tests are required; that `yarn test` passes is sufficient.
This commit is contained in:
Dandelion Mané 2019-05-19 14:17:46 +03:00 committed by GitHub
parent d2559960bb
commit 9d8d223653
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
17 changed files with 86 additions and 89 deletions

View File

@ -20,8 +20,7 @@ export const fallbackNodeType: NodeType = Object.freeze({
export const fallbackEdgeType: EdgeType = Object.freeze({ export const fallbackEdgeType: EdgeType = Object.freeze({
forwardName: "points to", forwardName: "points to",
backwardName: "is pointed to by", backwardName: "is pointed to by",
defaultForwardWeight: 1, defaultWeight: {forwards: 1, backwards: 1},
defaultBackwardWeight: 1,
prefix: EdgeAddress.empty, prefix: EdgeAddress.empty,
description: description:
"The fallback EdgeType for edges which don't have any other type", "The fallback EdgeType for edges which don't have any other type",

View File

@ -1,6 +1,7 @@
// @flow // @flow
import {type NodeAddressT, type EdgeAddressT} from "../core/graph"; import {type NodeAddressT, type EdgeAddressT} from "../core/graph";
import {type EdgeWeight} from "./weights";
/** /**
* This module defines `NodeType`s and `EdgeType`s, both of which are * This module defines `NodeType`s and `EdgeType`s, both of which are
@ -86,15 +87,10 @@ export type EdgeType = {|
// is "referenced by" // is "referenced by"
+backwardName: string, +backwardName: string,
// The default weight for the forward direction of this edge. // The default forwards and backwards weights for this edge.
// We use `1` as a default value ("of normal importance"). // We use `1` as a default value ("of normal importance").
// The weights have linear importance, i.e. 2 is twice as important as 1. // The weights have linear importance, i.e. 2 is twice as important as 1.
+defaultForwardWeight: number, +defaultWeight: EdgeWeight,
// The default weight for the backward direction of this edge.
// We use `1` as a default value ("of normal importance").
// The weights have linear importance, i.e. 2 is twice as important as 1.
+defaultBackwardWeight: number,
// The address that will be used to test whether an edge is a member // The address that will be used to test whether an edge is a member
// of this EdgeType. A given edge `e` is a member of the type `t` if // of this EdgeType. A given edge `e` is a member of the type `t` if

View File

@ -5,12 +5,26 @@ import {type NodeAddressT, type EdgeAddressT} from "../core/graph";
import type {EdgeType, NodeType} from "./types"; import type {EdgeType, NodeType} from "./types";
import type {PluginDeclaration} from "./pluginDeclaration"; import type {PluginDeclaration} from "./pluginDeclaration";
export type WeightedNodeType = {|+type: NodeType, +weight: number|}; /**
* Represents the weight for a particular Node (or NodeType).
* Weight 1 is the default value and signifies normal importance.
* Weights are linear, so 2 is twice as important as 1.
*/
export type NodeWeight = number;
/**
* Represents the forwards and backwards weights for a particular Edge (or
* EdgeType).
* Weight 1 is the default value and signifies normal importance.
* Weights are linear, so 2 is twice as important as 1.
*/
export type EdgeWeight = {|+forwards: number, +backwards: number|};
export type WeightedNodeType = {|+type: NodeType, +weight: NodeWeight|};
export type WeightedEdgeType = {| export type WeightedEdgeType = {|
+type: EdgeType, +type: EdgeType,
+forwardWeight: number, +weight: EdgeWeight,
+backwardWeight: number,
|}; |};
export type WeightedTypes = {| export type WeightedTypes = {|
@ -31,8 +45,7 @@ export function defaultWeightedNodeType(type: NodeType): WeightedNodeType {
export function defaultWeightedEdgeType(type: EdgeType): WeightedEdgeType { export function defaultWeightedEdgeType(type: EdgeType): WeightedEdgeType {
return { return {
type, type,
forwardWeight: type.defaultForwardWeight, weight: type.defaultWeight,
backwardWeight: type.defaultBackwardWeight,
}; };
} }

View File

@ -24,8 +24,8 @@ describe("analysis/weights", () => {
describe("defaultWeightedEdgeType", () => { describe("defaultWeightedEdgeType", () => {
it("sets default weights as specified in the type", () => { it("sets default weights as specified in the type", () => {
const wet = defaultWeightedEdgeType(assemblesEdgeType); const wet = defaultWeightedEdgeType(assemblesEdgeType);
expect(wet.forwardWeight).toEqual(wet.type.defaultForwardWeight); expect(wet.weight.forwards).toEqual(wet.type.defaultWeight.forwards);
expect(wet.backwardWeight).toEqual(wet.type.defaultBackwardWeight); expect(wet.weight.backwards).toEqual(wet.type.defaultWeight.backwards);
}); });
}); });
describe("defaultWeightsForDeclaration", () => { describe("defaultWeightsForDeclaration", () => {

View File

@ -15,8 +15,8 @@ export function weightsToEdgeEvaluator(
nodeTrie.add(type.prefix, weight); nodeTrie.add(type.prefix, weight);
} }
const edgeTrie = new EdgeTrie(); const edgeTrie = new EdgeTrie();
for (const {type, forwardWeight, backwardWeight} of weights.edges.values()) { for (const {type, weight} of weights.edges.values()) {
edgeTrie.add(type.prefix, {forwardWeight, backwardWeight}); edgeTrie.add(type.prefix, weight);
} }
function nodeWeight(n: NodeAddressT): number { function nodeWeight(n: NodeAddressT): number {
@ -28,10 +28,10 @@ export function weightsToEdgeEvaluator(
return function evaluator(edge: Edge) { return function evaluator(edge: Edge) {
const srcWeight = nodeWeight(edge.src); const srcWeight = nodeWeight(edge.src);
const dstWeight = nodeWeight(edge.dst); const dstWeight = nodeWeight(edge.dst);
const {forwardWeight, backwardWeight} = edgeTrie.getLast(edge.address); const {forwards, backwards} = edgeTrie.getLast(edge.address);
return { return {
toWeight: dstWeight * forwardWeight, toWeight: dstWeight * forwards,
froWeight: srcWeight * backwardWeight, froWeight: srcWeight * backwards,
}; };
}; };
} }

View File

@ -43,13 +43,17 @@ describe("analysis/weightsToEdgeEvaluator", () => {
const nodesMap = new Map(nodes.map((x) => [x.type.prefix, x])); const nodesMap = new Map(nodes.map((x) => [x.type.prefix, x]));
const edges = [ const edges = [
{ {
forwardWeight: NullUtil.orElse(assemblesForward, 1), weight: {
backwardWeight: NullUtil.orElse(assemblesBackward, 1), forwards: NullUtil.orElse(assemblesForward, 1),
backwards: NullUtil.orElse(assemblesBackward, 1),
},
type: assemblesEdgeType, type: assemblesEdgeType,
}, },
{ {
forwardWeight: NullUtil.orElse(baseForward, 1), weight: {
backwardWeight: NullUtil.orElse(baseBackward, 1), forwards: NullUtil.orElse(baseForward, 1),
backwards: NullUtil.orElse(baseBackward, 1),
},
type: fallbackEdgeType, type: fallbackEdgeType,
}, },
]; ];

View File

@ -206,8 +206,10 @@ describe("cli/pagerank", () => {
const edgeType: EdgeType = { const edgeType: EdgeType = {
forwardName: "bars", forwardName: "bars",
backwardName: "barred by", backwardName: "barred by",
defaultForwardWeight: 5, defaultWeight: {
defaultBackwardWeight: 3, forwards: 5,
backwards: 3,
},
prefix: EdgeAddress.fromParts(["hom"]), prefix: EdgeAddress.fromParts(["hom"]),
description: "an example edge type", description: "an example edge type",
}; };
@ -268,14 +270,9 @@ describe("cli/pagerank", () => {
expect(weightedNodeType.type).toEqual(nodeType); expect(weightedNodeType.type).toEqual(nodeType);
} }
for (const edgeType of declaration.edgeTypes) { for (const edgeType of declaration.edgeTypes) {
const weightedEdgeType = NullUtil.get(ws.edges.get(edgeType.prefix)); const {weight, type} = NullUtil.get(ws.edges.get(edgeType.prefix));
expect(weightedEdgeType.forwardWeight).toEqual( expect(weight).toEqual(edgeType.defaultWeight);
edgeType.defaultForwardWeight expect(type).toEqual(edgeType);
);
expect(weightedEdgeType.backwardWeight).toEqual(
edgeType.defaultBackwardWeight
);
expect(weightedEdgeType.type).toEqual(edgeType);
} }
} }
}); });

View File

@ -97,6 +97,8 @@ export type OrderedSparseMarkovChain = {|
+chain: SparseMarkovChain, +chain: SparseMarkovChain,
|}; |};
// TODO(@decentralion): Rename these fields to `forwards` and `backwards` to
// deduplicate with the EdgeWeight type defined by analysis/weights
export type EdgeWeight = {| export type EdgeWeight = {|
+toWeight: number, // weight from src to dst +toWeight: number, // weight from src to dst
+froWeight: number, // weight from dst to src +froWeight: number, // weight from dst to src

View File

@ -151,8 +151,7 @@ describe("explorer/pagerankTable/Aggregation", () => {
const edgeType: EdgeType = { const edgeType: EdgeType = {
forwardName: "marsellus", forwardName: "marsellus",
backwardName: "wallace", backwardName: "wallace",
defaultForwardWeight: 1, defaultWeight: {forwards: 1, backwards: 1},
defaultBackwardWeight: 1,
prefix: EdgeAddress.fromParts(["look", "like"]), prefix: EdgeAddress.fromParts(["look", "like"]),
description: "Connects example nodes for testing purposes.", description: "Connects example nodes for testing purposes.",
}; };

View File

@ -63,24 +63,21 @@ describe("explorer/pagerankTable/aggregate", () => {
foo: { foo: {
forwardName: "foos", forwardName: "foos",
backwardName: "foo'd by", backwardName: "foo'd by",
defaultForwardWeight: 1, defaultWeight: {forwards: 1, backwards: 1},
defaultBackwardWeight: 1,
prefix: EdgeAddress.fromParts(["foo"]), prefix: EdgeAddress.fromParts(["foo"]),
description: "Connects example foo edges.", description: "Connects example foo edges.",
}, },
bar: { bar: {
forwardName: "bars", forwardName: "bars",
backwardName: "bar'd by", backwardName: "bar'd by",
defaultForwardWeight: 1, defaultWeight: {forwards: 1, backwards: 1},
defaultBackwardWeight: 1,
prefix: EdgeAddress.fromParts(["bar"]), prefix: EdgeAddress.fromParts(["bar"]),
description: "Connects example bar edges.", description: "Connects example bar edges.",
}, },
empty: { empty: {
forwardName: "empty", forwardName: "empty",
backwardName: "emptied-by", backwardName: "emptied-by",
defaultForwardWeight: 1, defaultWeight: {forwards: 1, backwards: 1},
defaultBackwardWeight: 1,
prefix: EdgeAddress.empty, prefix: EdgeAddress.empty,
description: "Connects arbitrary edges.", description: "Connects arbitrary edges.",
}, },

View File

@ -10,27 +10,30 @@ export class EdgeTypeConfig extends React.Component<{
+onChange: (WeightedEdgeType) => void, +onChange: (WeightedEdgeType) => void,
}> { }> {
render() { render() {
const {weight, type} = this.props.weightedType;
const {forwards, backwards} = weight;
const {forwardName, backwardName, description} = type;
return ( return (
<div> <div>
<EdgeWeightSlider <EdgeWeightSlider
name={this.props.weightedType.type.backwardName} name={backwardName}
weight={this.props.weightedType.forwardWeight} weight={forwards}
description={this.props.weightedType.type.description} description={description}
onChange={(forwardWeight) => { onChange={(newForwards) => {
this.props.onChange({ this.props.onChange({
...this.props.weightedType, type,
forwardWeight, weight: {forwards: newForwards, backwards},
}); });
}} }}
/> />
<EdgeWeightSlider <EdgeWeightSlider
name={this.props.weightedType.type.forwardName} name={forwardName}
weight={this.props.weightedType.backwardWeight} weight={backwards}
description={this.props.weightedType.type.description} description={description}
onChange={(backwardWeight) => { onChange={(newBackwards) => {
this.props.onChange({ this.props.onChange({
...this.props.weightedType, type,
backwardWeight, weight: {forwards, backwards: newBackwards},
}); });
}} }}
/> />

View File

@ -15,8 +15,7 @@ describe("explorer/weights/EdgeTypeConfig", () => {
const onChange = jest.fn(); const onChange = jest.fn();
const wet = { const wet = {
type: assemblesEdgeType, type: assemblesEdgeType,
forwardWeight: 1, weight: {forwards: 1, backwards: 0.5},
backwardWeight: 0.5,
}; };
const element = shallow( const element = shallow(
<EdgeTypeConfig onChange={onChange} weightedType={wet} /> <EdgeTypeConfig onChange={onChange} weightedType={wet} />
@ -28,12 +27,12 @@ describe("explorer/weights/EdgeTypeConfig", () => {
it("sets up the forward weight slider", () => { it("sets up the forward weight slider", () => {
const {wet, forwardSlider} = example(); const {wet, forwardSlider} = example();
expect(forwardSlider.props().name).toBe(assemblesEdgeType.backwardName); expect(forwardSlider.props().name).toBe(assemblesEdgeType.backwardName);
expect(forwardSlider.props().weight).toBe(wet.forwardWeight); expect(forwardSlider.props().weight).toBe(wet.weight.forwards);
}); });
it("sets up the backward weight slider", () => { it("sets up the backward weight slider", () => {
const {wet, backwardSlider} = example(); const {wet, backwardSlider} = example();
expect(backwardSlider.props().name).toBe(assemblesEdgeType.forwardName); expect(backwardSlider.props().name).toBe(assemblesEdgeType.forwardName);
expect(backwardSlider.props().weight).toBe(wet.backwardWeight); expect(backwardSlider.props().weight).toBe(wet.weight.backwards);
}); });
it("has a description", () => { it("has a description", () => {
const {backwardSlider} = example(); const {backwardSlider} = example();
@ -44,14 +43,16 @@ describe("explorer/weights/EdgeTypeConfig", () => {
it("forward weight slider onChange works", () => { it("forward weight slider onChange works", () => {
const {wet, forwardSlider, onChange} = example(); const {wet, forwardSlider, onChange} = example();
forwardSlider.props().onChange(9); forwardSlider.props().onChange(9);
const updated = {...wet, forwardWeight: 9}; const weight = {...wet.weight, forwards: 9};
const updated = {...wet, weight};
expect(onChange).toHaveBeenCalledTimes(1); expect(onChange).toHaveBeenCalledTimes(1);
expect(onChange.mock.calls[0][0]).toEqual(updated); expect(onChange.mock.calls[0][0]).toEqual(updated);
}); });
it("backward weight slider onChange works", () => { it("backward weight slider onChange works", () => {
const {wet, backwardSlider, onChange} = example(); const {wet, backwardSlider, onChange} = example();
backwardSlider.props().onChange(9); backwardSlider.props().onChange(9);
const updated = {...wet, backwardWeight: 9}; const weight = {...wet.weight, backwards: 9};
const updated = {...wet, weight};
expect(onChange).toHaveBeenCalledTimes(1); expect(onChange).toHaveBeenCalledTimes(1);
expect(onChange.mock.calls[0][0]).toEqual(updated); expect(onChange.mock.calls[0][0]).toEqual(updated);
}); });

View File

@ -130,8 +130,7 @@ describe("explorer/weights/PluginWeightConfig", () => {
it("extra edge prefix in weighted types", () => { it("extra edge prefix in weighted types", () => {
const badTypes = defaultWeightsForAdapter(new FactorioStaticAdapter()); const badTypes = defaultWeightsForAdapter(new FactorioStaticAdapter());
badTypes.edges.set(fallbackEdgeType.prefix, { badTypes.edges.set(fallbackEdgeType.prefix, {
forwardWeight: 5, weight: {forwards: 5, backwards: 3},
backwardWeight: 3,
type: fallbackEdgeType, type: fallbackEdgeType,
}); });
checkError(badTypes); checkError(badTypes);

View File

@ -24,18 +24,16 @@ export const machineNodeType: NodeType = Object.freeze({
export const assemblesEdgeType: EdgeType = Object.freeze({ export const assemblesEdgeType: EdgeType = Object.freeze({
forwardName: "assembles", forwardName: "assembles",
defaultForwardWeight: 2,
backwardName: "is assembled by", backwardName: "is assembled by",
defaultBackwardWeight: 2 ** -2, defaultWeight: {forwards: 2, backwards: 2 ** -2},
prefix: EdgeAddress.fromParts(["factorio", "assembles"]), prefix: EdgeAddress.fromParts(["factorio", "assembles"]),
description: "Connects assembly machines to products they assemble.", description: "Connects assembly machines to products they assemble.",
}); });
export const transportsEdgeType: EdgeType = Object.freeze({ export const transportsEdgeType: EdgeType = Object.freeze({
forwardName: "transports", forwardName: "transports",
defaultForwardWeight: 1,
backwardName: "is transported by", backwardName: "is transported by",
defaultBackwardWeight: 2 ** -1, defaultWeight: {forwards: 2, backwards: 2 ** -1},
prefix: EdgeAddress.fromParts(["factorio", "transports"]), prefix: EdgeAddress.fromParts(["factorio", "transports"]),
description: "Connects transporter belts to objects they transport.", description: "Connects transporter belts to objects they transport.",
}); });

View File

@ -17,8 +17,7 @@ const hasParentEdgeType = Object.freeze({
forwardName: "has parent", forwardName: "has parent",
backwardName: "is parent of", backwardName: "is parent of",
prefix: E.Prefix.hasParent, prefix: E.Prefix.hasParent,
defaultForwardWeight: 1, defaultWeight: {forwards: 1, backwards: 1},
defaultBackwardWeight: 1,
description: "Connects a Git commit to its parent commit(s).", description: "Connects a Git commit to its parent commit(s).",
}); });

View File

@ -74,8 +74,7 @@ const nodeTypes = Object.freeze([
const authorsEdgeType = Object.freeze({ const authorsEdgeType = Object.freeze({
forwardName: "authors", forwardName: "authors",
backwardName: "is authored by", backwardName: "is authored by",
defaultForwardWeight: 1 / 2, defaultWeight: {forwards: 1 / 2, backwards: 1},
defaultBackwardWeight: 1,
prefix: E.Prefix.authors, prefix: E.Prefix.authors,
description: dedent`\ description: dedent`\
Connects a GitHub account to a post that they authored. Connects a GitHub account to a post that they authored.
@ -87,8 +86,7 @@ const authorsEdgeType = Object.freeze({
const hasParentEdgeType = Object.freeze({ const hasParentEdgeType = Object.freeze({
forwardName: "has parent", forwardName: "has parent",
backwardName: "has child", backwardName: "has child",
defaultForwardWeight: 1, defaultWeight: {forwards: 1, backwards: 1 / 4},
defaultBackwardWeight: 1 / 4,
prefix: E.Prefix.hasParent, prefix: E.Prefix.hasParent,
description: dedent`\ description: dedent`\
Connects a GitHub entity to its child entities. Connects a GitHub entity to its child entities.
@ -101,8 +99,7 @@ const hasParentEdgeType = Object.freeze({
const mergedAsEdgeType = Object.freeze({ const mergedAsEdgeType = Object.freeze({
forwardName: "merges", forwardName: "merges",
backwardName: "is merged by", backwardName: "is merged by",
defaultForwardWeight: 1 / 2, defaultWeight: {forwards: 1 / 2, backwards: 1},
defaultBackwardWeight: 1,
prefix: E.Prefix.mergedAs, prefix: E.Prefix.mergedAs,
description: dedent`\ description: dedent`\
Connects a GitHub pull request to the Git commit that it merges. Connects a GitHub pull request to the Git commit that it merges.
@ -112,8 +109,7 @@ const mergedAsEdgeType = Object.freeze({
const referencesEdgeType = Object.freeze({ const referencesEdgeType = Object.freeze({
forwardName: "references", forwardName: "references",
backwardName: "is referenced by", backwardName: "is referenced by",
defaultForwardWeight: 1, defaultWeight: {forwards: 1, backwards: 0},
defaultBackwardWeight: 0,
prefix: E.Prefix.references, prefix: E.Prefix.references,
description: dedent`\ description: dedent`\
Connects a GitHub post to an entity that it references. Connects a GitHub post to an entity that it references.
@ -127,8 +123,7 @@ const referencesEdgeType = Object.freeze({
const mentionsAuthorEdgeType = Object.freeze({ const mentionsAuthorEdgeType = Object.freeze({
forwardName: "mentions author of", forwardName: "mentions author of",
backwardName: "has author mentioned by", backwardName: "has author mentioned by",
defaultForwardWeight: 1, defaultWeight: {forwards: 1, backwards: 0},
defaultBackwardWeight: 0,
prefix: E.Prefix.mentionsAuthor, prefix: E.Prefix.mentionsAuthor,
description: dedent`\ description: dedent`\
Connects a post that mentions a user to posts in the same thread that Connects a post that mentions a user to posts in the same thread that
@ -144,8 +139,7 @@ const mentionsAuthorEdgeType = Object.freeze({
const reactsHeartEdgeType = Object.freeze({ const reactsHeartEdgeType = Object.freeze({
forwardName: "reacted ❤️ to", forwardName: "reacted ❤️ to",
backwardName: "got ❤️ from", backwardName: "got ❤️ from",
defaultForwardWeight: 2, defaultWeight: {forwards: 2, backwards: 0},
defaultBackwardWeight: 0,
prefix: E.Prefix.reactsHeart, prefix: E.Prefix.reactsHeart,
description: dedent`\ description: dedent`\
Connects users to posts to which they gave a reaction. Connects users to posts to which they gave a reaction.
@ -155,8 +149,7 @@ const reactsHeartEdgeType = Object.freeze({
const reactsThumbsUpEdgeType = Object.freeze({ const reactsThumbsUpEdgeType = Object.freeze({
forwardName: "reacted 👍 to", forwardName: "reacted 👍 to",
backwardName: "got 👍 from", backwardName: "got 👍 from",
defaultForwardWeight: 1, defaultWeight: {forwards: 1, backwards: 0},
defaultBackwardWeight: 0,
prefix: E.Prefix.reactsThumbsUp, prefix: E.Prefix.reactsThumbsUp,
description: dedent`\ description: dedent`\
Connects users to posts to which they gave a 👍 reaction. Connects users to posts to which they gave a 👍 reaction.
@ -166,8 +159,7 @@ const reactsThumbsUpEdgeType = Object.freeze({
const reactsHoorayEdgeType = Object.freeze({ const reactsHoorayEdgeType = Object.freeze({
forwardName: "reacted 🎉 to", forwardName: "reacted 🎉 to",
backwardName: "got 🎉 from", backwardName: "got 🎉 from",
defaultForwardWeight: 4, defaultWeight: {forwards: 4, backwards: 0},
defaultBackwardWeight: 0,
prefix: E.Prefix.reactsHooray, prefix: E.Prefix.reactsHooray,
description: dedent`\ description: dedent`\
Connects users to posts to which they gave a 🎉 reaction. Connects users to posts to which they gave a 🎉 reaction.
@ -177,8 +169,7 @@ const reactsHoorayEdgeType = Object.freeze({
const reactsRocketEdgeType = Object.freeze({ const reactsRocketEdgeType = Object.freeze({
forwardName: "reacted 🚀 to", forwardName: "reacted 🚀 to",
backwardName: "got 🚀 from", backwardName: "got 🚀 from",
defaultForwardWeight: 1, defaultWeight: {forwards: 1, backwards: 0},
defaultBackwardWeight: 0,
prefix: E.Prefix.reactsRocket, prefix: E.Prefix.reactsRocket,
description: dedent`\ description: dedent`\
Connects users to posts to which they gave a 🚀 reaction. Connects users to posts to which they gave a 🚀 reaction.

View File

@ -64,8 +64,7 @@ const dependsOnEdgeType = Object.freeze({
forwardName: "depends on", forwardName: "depends on",
backwardName: "is depended on by", backwardName: "is depended on by",
prefix: EdgeAddress.append(EDGE_PREFIX, "DEPENDS_ON"), prefix: EdgeAddress.append(EDGE_PREFIX, "DEPENDS_ON"),
defaultForwardWeight: 1, defaultWeight: {forwards: 1, backwards: 0},
defaultBackwardWeight: 0,
description: "Generic edge for flowing credit in the Odyssey plugin", description: "Generic edge for flowing credit in the Odyssey plugin",
}); });