createWeightedGraph = () => {
const graph = new sc.core.graph.Graph();
const weights = sc.core.weights.empty();
for (const meal of mockApi.getMeals()) {
graph.addNode({
address: sc.core.graph.NodeAddress.append(
declaration.nodePrefix,
"MEAL",
meal.id
),
description: `A beautiful ${meal.name}`,
timestampMs: meal.timestamp
});
const chef = mockApi.getChefById(meal.chefId);
graph.addNode({
address: sc.core.graph.NodeAddress.append(
declaration.nodePrefix,
"CHEF",
chef.id
),
description: `A chef named ${chef.name}`,
timestampMs: null
});
const table = mockApi.getTableById(meal.tableId);
graph.addNode({
address: sc.core.graph.NodeAddress.append(
declaration.nodePrefix,
"TABLE",
table.id
),
description: `Sturdy table ${table.id}`,
timestampMs: null
});
const server = mockApi.getServerById(table.serverId);
graph.addNode({
address: sc.core.graph.NodeAddress.append(
declaration.nodePrefix,
"SERVER",
server.id
),
description: `A server named ${server.name}`,
timestampMs: null
});
// We've made all the important nodes for this meal,
// now let's connect them with edges!
graph.addEdge({
address: sc.core.graph.EdgeAddress.append(
declaration.edgePrefix, // Notice, we're using EdgeAddress and edgePrefix here!
"COOKED",
chef.id,
meal.id
),
timestamp: meal.timestamp,
// Pro tip: Helper functions for constructing each address type from an id/object can be really useful
// for reducing bugs and duplicate code. Otherwise, we'll be doing a lot of this:
src: sc.core.graph.NodeAddress.append(
declaration.nodePrefix,
"CHEF",
chef.id
),
// Make sure the src -> dst match the "forwardName" semantics in the declaration.
dst: sc.core.graph.NodeAddress.append(
declaration.nodePrefix,
"MEAL",
meal.id
)
});
graph.addEdge({
address: sc.core.graph.EdgeAddress.append(
declaration.edgePrefix,
"DELIVERED_TO",
meal.id,
table.id
),
timestamp: meal.timestamp, // We don't know exactly when it was delivered, so we're approximating.
src: sc.core.graph.NodeAddress.append(
declaration.nodePrefix,
"MEAL",
meal.id
),
dst: sc.core.graph.NodeAddress.append(
declaration.nodePrefix,
"TABLE",
table.id
)
});
graph.addEdge({
address: sc.core.graph.EdgeAddress.append(
declaration.edgePrefix,
"SERVED",
server.id,
table.id
),
timestamp: null,
src: sc.core.graph.NodeAddress.append(
declaration.nodePrefix,
"SERVER",
server.id
),
dst: sc.core.graph.NodeAddress.append(
declaration.nodePrefix,
"TABLE",
table.id
)
});
// Whew! We're almost there.
// If you want to get fancy, you can create custom per-node or per-edge weights
// by mapping addresses to weights.
weights.nodeWeights.set(
sc.core.graph.NodeAddress.append(declaration.nodePrefix, "MEAL", meal.id),
// This will make the weight of each meal directly proportional to "how much money it costs."
meal.cost
);
// Let's get real fancy and add some logic!
if (chef.rank === "JUNIOR") {
// Depending on how you created your address declarations, you can also add weight multipliers
// to node or edge address prefixes. This might affect how you want to order address parts.
// See how this is just a prefix which will match all COOKED edges for the current chef.
weights.edgeWeights.set(
sc.core.graph.EdgeAddress.append(
declaration.edgePrefix,
"COOKED",
chef.id
),
{
// Since we are adding a prefix MULTIPLIER, we set 1 to indicate "no change"
forwards: 1,
// Semantically, reducing the edge weight here means less cred will flow from meals to JUNIOR chefs.
// Instead, more cred will flow out from other edges, in this case from meals to servers.
// Maybe a little strange semantically for servers to get more cred when serving meals made
// by junior chefs, so I might not build the graph this way in a real scenario. Then again, the
// servers have to deal with unhappy customers when the food is worse, so maybe this does make
// sense! The lesson is, although some configurability is given to the Instance Admins through
// the Declaration, we as devs have a LOT of power to influence cred by how we choose to
// build and alter graphs.
backwards: 1 / 2
}
);
}
// I created Nodes, then Edges, then Weights, but you can mix it up and add
// things however makes sense for the APIs and data you're working with.
}
return { graph, weights };
}