269 lines
5.5 KiB
JavaScript
269 lines
5.5 KiB
JavaScript
|
"use strict";
|
||
|
|
||
|
const fs = require('fs');
|
||
|
const {EOL} = require('os');
|
||
|
const eps = 1e-6;
|
||
|
|
||
|
function chooseActiveNodeForRelabel(c, e, h) {
|
||
|
let n = h.length;
|
||
|
for (let i = 0; i < n; i++) {
|
||
|
if (e[i] <= 0) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
for (let j = 0; j < n; j++) {
|
||
|
if (c[i][j] <= 0) {
|
||
|
// in edge E_f
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if (h[i] > h[j]) {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return i;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
function chooseActiveNodeForPush(c, e, h) {
|
||
|
let n = h.length;
|
||
|
for (let i = 0; i < n; i++) { // what about replacing it with priority queues
|
||
|
if (e[i] <= 0) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// we are looking for nodes if excess
|
||
|
|
||
|
for (let j = 0; j < n; j++) { // what about replacing it with priority queues
|
||
|
if (c[i][j] <= 0 || h[i] != h[j] + 1) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// c[i][j] > 0 && h[i] == h[j] + 1
|
||
|
return [i, j];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return [-1, -1];
|
||
|
}
|
||
|
|
||
|
function findMaximumFlow(graph) {
|
||
|
let nodes = [];
|
||
|
let m = {};
|
||
|
for (let el of graph.nodes) {
|
||
|
nodes.push(el);
|
||
|
}
|
||
|
|
||
|
for (let i = 0; i < nodes.length; i++) {
|
||
|
m[nodes[i]] = i;
|
||
|
}
|
||
|
|
||
|
const n = nodes.length;
|
||
|
let c = new Array(n);
|
||
|
let oc = new Array(n);
|
||
|
let f = new Array(n);
|
||
|
let h = new Array(n);
|
||
|
let e = new Array(n); // excess flow
|
||
|
for (let i = 0; i < n; i++) {
|
||
|
c[i] = new Array(n); // residual capacity
|
||
|
oc[i] = new Array(n); // residual capacity
|
||
|
f[i] = new Array(n); // flow
|
||
|
h[i] = 0; // node height
|
||
|
e[i] = 0; // node flow excess
|
||
|
}
|
||
|
|
||
|
h[m[graph.source]] = n;
|
||
|
|
||
|
for (let i = 0; i < n; i++) {
|
||
|
for (let j = 0; j < n; j++) {
|
||
|
f[i][j] = 0;
|
||
|
c[i][j] = 0;
|
||
|
oc[i][j] = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// init capacity
|
||
|
|
||
|
for (let v in graph.edges) {
|
||
|
for (let {node, weight} of graph.edges[v]) {
|
||
|
c[m[v]][m[node]] = weight;
|
||
|
oc[m[v]][m[node]] = weight;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
for (let {node, weight} of graph.edges[graph.source]) {
|
||
|
f[m[graph.source]][m[node]] = weight;
|
||
|
e[m[node]] = weight;
|
||
|
e[m[graph.source]] -= weight;
|
||
|
}
|
||
|
|
||
|
console.log('Capacity', c);
|
||
|
console.log('Flow', f);
|
||
|
console.log('Height', h);
|
||
|
console.log('Excess', e);
|
||
|
|
||
|
// try push
|
||
|
// if not relabel
|
||
|
// and repeat till it's possible
|
||
|
|
||
|
// quite naive way to choose active node for push
|
||
|
do {
|
||
|
// push
|
||
|
let [ci, cj] = chooseActiveNodeForPush(c, e, h);
|
||
|
if (ci >= 0 && cj >= 0) {
|
||
|
console.log(`Pushing (${ci},${cj})`);
|
||
|
let flow = Math.min(e[ci], c[ci][cj]);
|
||
|
if (oc[ci][cj] > 0) { // if ci, cj exists in e edges
|
||
|
f[ci][cj] += flow;
|
||
|
} else {
|
||
|
f[cj][ci] -= flow;
|
||
|
}
|
||
|
|
||
|
e[ci] -= flow;
|
||
|
e[cj] += flow;
|
||
|
} else {
|
||
|
// relabel
|
||
|
let ri = chooseActiveNodeForRelabel(c, e, h);
|
||
|
if (ri >= 0) {
|
||
|
console.log(`Relabeling (${ri})`);
|
||
|
let hMax = -Infinity;
|
||
|
for (let i = 0; i < n; i++) {
|
||
|
if (c[ri][i] <= 0) {
|
||
|
continue;
|
||
|
}
|
||
|
hMax = Math.max(hMax, h[i]);
|
||
|
}
|
||
|
if (hMax <= 0) {
|
||
|
continue;
|
||
|
}
|
||
|
h[ri] = hMax + 1;
|
||
|
console.log(`Setting height ${h[ri]} for ${ri}`);
|
||
|
} else {
|
||
|
console.log('Maximal flow...');
|
||
|
break; // we have maximal flow
|
||
|
}
|
||
|
}
|
||
|
} while (true);
|
||
|
|
||
|
let flow = 0;
|
||
|
for (let j = 0; j < n; j++) {
|
||
|
flow += f[m[graph.source]][j];
|
||
|
}
|
||
|
return flow;
|
||
|
}
|
||
|
|
||
|
function sourceCmdLine(ctx, cmdLine) {
|
||
|
ctx.source = cmdLine.slice(
|
||
|
'source'.length,
|
||
|
cmdLine.indexOf(';')
|
||
|
).trim();
|
||
|
}
|
||
|
|
||
|
function targetCmdLine(ctx, cmdLine) {
|
||
|
ctx.target = cmdLine.slice(
|
||
|
'target'.length,
|
||
|
cmdLine.indexOf(';')
|
||
|
).trim();
|
||
|
}
|
||
|
|
||
|
function edgeCmdLine(ctx, cmdLine) {
|
||
|
let edgeLine = cmdLine.slice(
|
||
|
'edge'.length,
|
||
|
cmdLine.indexOf(';')
|
||
|
).trim();
|
||
|
console.log(`Edge line ${edgeLine}`);
|
||
|
|
||
|
const matches = edgeLine.match(/^([a-zA-Z]*) -> ([a-zA-Z]*): ([0-9.]*)$/);
|
||
|
// console.log('Matches', matches);
|
||
|
const v = matches[1];
|
||
|
const z = matches[2];
|
||
|
const weight = Number.parseFloat(matches[3]);
|
||
|
|
||
|
console.log(`Parsed ${v} -> ${z} : ${weight}`);
|
||
|
|
||
|
ctx.nodes.add(v);
|
||
|
ctx.nodes.add(z);
|
||
|
let targetEdges = ctx.edges[v] || [];
|
||
|
targetEdges.push({node: z, weight});
|
||
|
ctx.edges[v] = targetEdges;
|
||
|
|
||
|
}
|
||
|
|
||
|
function processCmdLine(ctx, cmdLine) {
|
||
|
console.log(`Processing command line ${cmdLine}`);
|
||
|
if (cmdLine.startsWith('source')) {
|
||
|
sourceCmdLine(ctx, cmdLine);
|
||
|
} else if (cmdLine.startsWith('target')) {
|
||
|
targetCmdLine(ctx, cmdLine);
|
||
|
} else if (cmdLine.startsWith('edge')) {
|
||
|
edgeCmdLine(ctx, cmdLine);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
console.log(`Process argv: ${process.argv}`);
|
||
|
|
||
|
const inputPath = process.argv.length > 2 ? process.argv[2] : '-';
|
||
|
const outputPath = process.argv.length > 3 ? process.argv[3] : '-';
|
||
|
|
||
|
console.log(`Input path ${inputPath}, output path: ${outputPath}`);
|
||
|
|
||
|
const inputStream = inputPath === '-'
|
||
|
? process.stdin
|
||
|
: fs.createReadStream(inputPath);
|
||
|
|
||
|
// const outputStream = outputPath === '-'
|
||
|
// ? process.stdout
|
||
|
// : fs.createWriteStream(outputStream);
|
||
|
|
||
|
let inputBuffer = "";
|
||
|
let ctx = {
|
||
|
source: '',
|
||
|
target: '',
|
||
|
edges: {},
|
||
|
nodes: new Set()
|
||
|
};
|
||
|
|
||
|
function processChunk(chunk) {
|
||
|
inputBuffer += chunk;
|
||
|
console.log(`Chunk: ${chunk}`);
|
||
|
let eolIdx;
|
||
|
do {
|
||
|
eolIdx = inputBuffer.indexOf(EOL);
|
||
|
if (eolIdx < 0 && !chunk) {
|
||
|
eolIdx = inputBuffer.length;
|
||
|
}
|
||
|
|
||
|
let cmdLine = null;
|
||
|
if (eolIdx >= 0) {
|
||
|
console.log(`We got new line ${eolIdx}`);
|
||
|
cmdLine = inputBuffer.slice(0, eolIdx);
|
||
|
processCmdLine(ctx, cmdLine);
|
||
|
inputBuffer = inputBuffer.slice(eolIdx+EOL.length);
|
||
|
}
|
||
|
} while (eolIdx >= 0 && inputBuffer.length);
|
||
|
}
|
||
|
|
||
|
inputStream.on('data', (chunk) => {
|
||
|
processChunk(chunk);
|
||
|
});
|
||
|
|
||
|
inputStream.on('end', (chunk) => {
|
||
|
console.log(`end chunk ${chunk}`);
|
||
|
processChunk(chunk);
|
||
|
console.log('Got following graph', ctx);
|
||
|
|
||
|
for (let v in ctx.edges) {
|
||
|
for (let {node, weight} of ctx.edges[v]) {
|
||
|
console.log(`${v} ${node} : ${weight}`);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
let mflow = findMaximumFlow(ctx);
|
||
|
|
||
|
console.log(`Maximum flow is ${mflow}`);
|
||
|
|
||
|
});
|