{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Minimum cost: 150\n",
      "\n",
      " Arc    Flow / Capacity Cost\n",
      "0 -> 1   12  /  15       48\n",
      "0 -> 2    8  /   8       32\n",
      "1 -> 2    8  /  20       16\n",
      "1 -> 3    4  /   4       8\n",
      "1 -> 4    0  /  10       0\n",
      "2 -> 3   12  /  15       12\n",
      "2 -> 4    4  /   4       12\n",
      "3 -> 4   11  /  20       22\n",
      "4 -> 2    0  /   5       0\n"
     ]
    }
   ],
   "source": [
    "\"\"\"From Bradley, Hax and Maganti, 'Applied Mathematical Programming', figure 8.1.\"\"\"\n",
    "import numpy as np\n",
    "\n",
    "from ortools.graph.python import min_cost_flow\n",
    "\n",
    "\n",
    "def main():\n",
    "    \"\"\"MinCostFlow simple interface example.\"\"\"\n",
    "    # Instantiate a SimpleMinCostFlow solver.\n",
    "    smcf = min_cost_flow.SimpleMinCostFlow()\n",
    "\n",
    "    # Define four parallel arrays: sources, destinations, capacities,\n",
    "    # and unit costs between each pair. For instance, the arc from node 0\n",
    "    # to node 1 has a capacity of 15.\n",
    "    start_nodes = np.array([0, 0, 1, 1, 1, 2, 2, 3, 4])\n",
    "    end_nodes = np.array([1, 2, 2, 3, 4, 3, 4, 4, 2])\n",
    "    capacities = np.array([15, 8, 20, 4, 10, 15, 4, 20, 5])\n",
    "    unit_costs = np.array([4, 4, 2, 2, 6, 1, 3, 2, 3])\n",
    "\n",
    "    # Define an array of supplies at each node.\n",
    "    supplies = [20, 0, 0, -5, -15]\n",
    "\n",
    "    # Add arcs, capacities and costs in bulk using numpy.\n",
    "    all_arcs = smcf.add_arcs_with_capacity_and_unit_cost(\n",
    "        start_nodes, end_nodes, capacities, unit_costs\n",
    "    )\n",
    "\n",
    "    # Add supply for each nodes.\n",
    "    smcf.set_nodes_supplies(np.arange(0, len(supplies)), supplies)\n",
    "\n",
    "    # Find the min cost flow.\n",
    "    status = smcf.solve()\n",
    "\n",
    "    if status != smcf.OPTIMAL:\n",
    "        print(\"There was an issue with the min cost flow input.\")\n",
    "        print(f\"Status: {status}\")\n",
    "        exit(1)\n",
    "    print(f\"Minimum cost: {smcf.optimal_cost()}\")\n",
    "    print(\"\")\n",
    "    print(\" Arc    Flow / Capacity Cost\")\n",
    "    solution_flows = smcf.flows(all_arcs)\n",
    "    costs = solution_flows * unit_costs\n",
    "    for arc, flow, cost in zip(all_arcs, solution_flows, costs):\n",
    "        print(\n",
    "            f\"{smcf.tail(arc):1} -> {smcf.head(arc)}  {flow:3}  / {smcf.capacity(arc):3}       {cost}\"\n",
    "        )\n",
    "\n",
    "\n",
    "if __name__ == \"__main__\":\n",
    "    main()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "ph1",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.10.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
