123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228 |
- 'use strict';
- // Load modules
- const Hoek = require('hoek');
- // Declare internals
- const internals = {};
- module.exports = class Topo {
- constructor() {
- this._items = [];
- this.nodes = [];
- }
- add(nodes, options) {
- options = options || {};
- // Validate rules
- const before = [].concat(options.before || []);
- const after = [].concat(options.after || []);
- const group = options.group || '?';
- const sort = options.sort || 0; // Used for merging only
- Hoek.assert(!before.includes(group), `Item cannot come before itself: ${group}`);
- Hoek.assert(!before.includes('?'), 'Item cannot come before unassociated items');
- Hoek.assert(!after.includes(group), `Item cannot come after itself: ${group}`);
- Hoek.assert(!after.includes('?'), 'Item cannot come after unassociated items');
- ([].concat(nodes)).forEach((node, i) => {
- const item = {
- seq: this._items.length,
- sort,
- before,
- after,
- group,
- node
- };
- this._items.push(item);
- });
- // Insert event
- const error = this._sort();
- Hoek.assert(!error, 'item', (group !== '?' ? `added into group ${group}` : ''), 'created a dependencies error');
- return this.nodes;
- }
- merge(others) {
- others = [].concat(others);
- for (let i = 0; i < others.length; ++i) {
- const other = others[i];
- if (other) {
- for (let j = 0; j < other._items.length; ++j) {
- const item = Object.assign({}, other._items[j]); // Shallow cloned
- this._items.push(item);
- }
- }
- }
- // Sort items
- this._items.sort(internals.mergeSort);
- for (let i = 0; i < this._items.length; ++i) {
- this._items[i].seq = i;
- }
- const error = this._sort();
- Hoek.assert(!error, 'merge created a dependencies error');
- return this.nodes;
- }
- _sort() {
- // Construct graph
- const graph = {};
- const graphAfters = Object.create(null); // A prototype can bungle lookups w/ false positives
- const groups = Object.create(null);
- for (let i = 0; i < this._items.length; ++i) {
- const item = this._items[i];
- const seq = item.seq; // Unique across all items
- const group = item.group;
- // Determine Groups
- groups[group] = groups[group] || [];
- groups[group].push(seq);
- // Build intermediary graph using 'before'
- graph[seq] = item.before;
- // Build second intermediary graph with 'after'
- const after = item.after;
- for (let j = 0; j < after.length; ++j) {
- graphAfters[after[j]] = (graphAfters[after[j]] || []).concat(seq);
- }
- }
- // Expand intermediary graph
- let graphNodes = Object.keys(graph);
- for (let i = 0; i < graphNodes.length; ++i) {
- const node = graphNodes[i];
- const expandedGroups = [];
- const graphNodeItems = Object.keys(graph[node]);
- for (let j = 0; j < graphNodeItems.length; ++j) {
- const group = graph[node][graphNodeItems[j]];
- groups[group] = groups[group] || [];
- for (let k = 0; k < groups[group].length; ++k) {
- expandedGroups.push(groups[group][k]);
- }
- }
- graph[node] = expandedGroups;
- }
- // Merge intermediary graph using graphAfters into final graph
- const afterNodes = Object.keys(graphAfters);
- for (let i = 0; i < afterNodes.length; ++i) {
- const group = afterNodes[i];
- if (groups[group]) {
- for (let j = 0; j < groups[group].length; ++j) {
- const node = groups[group][j];
- graph[node] = graph[node].concat(graphAfters[group]);
- }
- }
- }
- // Compile ancestors
- let children;
- const ancestors = {};
- graphNodes = Object.keys(graph);
- for (let i = 0; i < graphNodes.length; ++i) {
- const node = graphNodes[i];
- children = graph[node];
- for (let j = 0; j < children.length; ++j) {
- ancestors[children[j]] = (ancestors[children[j]] || []).concat(node);
- }
- }
- // Topo sort
- const visited = {};
- const sorted = [];
- for (let i = 0; i < this._items.length; ++i) { // Really looping thru item.seq values out of order
- let next = i;
- if (ancestors[i]) {
- next = null;
- for (let j = 0; j < this._items.length; ++j) { // As above, these are item.seq values
- if (visited[j] === true) {
- continue;
- }
- if (!ancestors[j]) {
- ancestors[j] = [];
- }
- const shouldSeeCount = ancestors[j].length;
- let seenCount = 0;
- for (let k = 0; k < shouldSeeCount; ++k) {
- if (visited[ancestors[j][k]]) {
- ++seenCount;
- }
- }
- if (seenCount === shouldSeeCount) {
- next = j;
- break;
- }
- }
- }
- if (next !== null) {
- visited[next] = true;
- sorted.push(next);
- }
- }
- if (sorted.length !== this._items.length) {
- return new Error('Invalid dependencies');
- }
- const seqIndex = {};
- for (let i = 0; i < this._items.length; ++i) {
- const item = this._items[i];
- seqIndex[item.seq] = item;
- }
- const sortedNodes = [];
- this._items = sorted.map((value) => {
- const sortedItem = seqIndex[value];
- sortedNodes.push(sortedItem.node);
- return sortedItem;
- });
- this.nodes = sortedNodes;
- }
- };
- internals.mergeSort = (a, b) => {
- return a.sort === b.sort ? 0 : (a.sort < b.sort ? -1 : 1);
- };
|