function graphTopoSort(numberNodes, edges) {
  const nodes = new Map();
  const order = [];
  const queue = [];
  for (let i = 0; i < numberNodes; i++) {
    nodes.set(i, {in: 0, out: new Set()});
  }

  edges.forEach(edge => {
    const [node_id, pre_id] = edge;
    nodes.get(node_id).in += 1;
    nodes.get(pre_id).out.add(node_id);
  });

  for (let [node_id, value] of nodes.entries()) {
    if (value.in === 0) {
      queue.push(node_id);
    }
  }

  while (queue.length) {
    const node_id = queue.shift();
    for (let outgoing_id of nodes.get(node_id).out) {
      nodes.get(outgoing_id).in -= 1;
      if (nodes.get(outgoing_id).in === 0) {
        queue.push(outgoing_id);
      }
    }
    order.push(node_id);
  }

  return order.length == numberNodes ? order : [];
}

console.log(graphTopoSort(3, [[0, 1], [0, 2]]));