1
0
Fork 0
maximum-weight-matching/tests/generate/make_random_graph.py

136 lines
3.6 KiB
Python
Raw Normal View History

2023-02-11 22:39:46 +01:00
#!/usr/bin/env python3
"""
Generate a random graph in DIMACS format.
"""
from __future__ import annotations
import sys
import argparse
import random
from typing import TextIO
def write_dimacs_graph(
f: TextIO,
2024-06-22 20:04:49 +02:00
edges: list[tuple[int, int, float]]
2023-02-11 22:39:46 +01:00
) -> None:
"""Write a graph in DIMACS edge list format."""
num_vertex = 1 + max(max(x, y) for (x, y, _w) in edges)
num_edge = len(edges)
print(f"p edge {num_vertex} {num_edge}", file=f)
integer_weights = all(isinstance(w, int) for (_x, _y, w) in edges)
for (x, y, w) in edges:
if integer_weights:
print(f"e {x+1} {y+1} {w}", file=f)
else:
print(f"e {x+1} {y+1} {w:.12g}", file=f)
def make_random_graph(
n: int,
m: int,
max_weight: float,
float_weights: bool,
rng: random.Random
2024-06-22 20:04:49 +02:00
) -> list[tuple[int, int, float]]:
2023-02-11 22:39:46 +01:00
"""Generate a random graph with random edge weights."""
edge_set: set[tuple[int, int]] = set()
if 3 * m < n * (n - 2) // 2:
# Simply add random edges until we have enough.
while len(edge_set) < m:
x = rng.randint(0, n - 1)
y = rng.randint(0, n - 2)
if y < x:
(x, y) = (y, x)
else:
y += 1
2023-02-11 22:39:46 +01:00
edge_set.add((x, y))
else:
# We need a very dense graph.
# Generate all edge candidates and choose a random subset.
edge_candidates = [(x, y)
for x in range(n - 1)
for y in range(x + 1, n)]
rng.shuffle(edge_candidates)
edge_set.update(edge_candidates[:m])
2024-06-22 20:04:49 +02:00
edges: list[tuple[int, int, float]] = []
2023-02-11 22:39:46 +01:00
for (x, y) in sorted(edge_set):
2024-06-22 20:04:49 +02:00
w: float
2023-02-11 22:39:46 +01:00
if float_weights:
w = rng.uniform(1.0e-8, max_weight)
else:
w = rng.randint(1, int(max_weight))
edges.append((x, y, w))
return edges
def main() -> int:
"""Main program."""
parser = argparse.ArgumentParser()
parser.description = "Generate a random graph in DIMACS format."
parser.add_argument("--seed",
action="store",
type=int,
help="random seed")
parser.add_argument("--maxweight",
action="store",
type=float,
default=1000000,
help="maximum edge weight")
parser.add_argument("--float",
action="store_true",
help="use floating point edge weights")
parser.add_argument("n",
action="store",
type=int,
help="number of vertices")
parser.add_argument("m",
action="store",
type=int,
help="number of edges")
args = parser.parse_args()
if args.n < 2:
print("ERROR: Number of vertices must be >= 2", file=sys.stderr)
return 1
if args.m < 1:
print("ERROR: Number of edges must be >= 1", file=sys.stderr)
return 1
if args.m > args.n * (args.n - 1) // 2:
print("ERROR: Too many edges", file=sys.stderr)
return 1
if args.maxweight < 1.0e-6:
print("ERROR: Invalid maximum edge weight", file=sys.stderr)
return 1
if args.seed is None:
rng = random.Random()
else:
rng = random.Random(args.seed)
edges = make_random_graph(args.n, args.m, args.maxweight, args.float, rng)
write_dimacs_graph(sys.stdout, edges)
return 0
if __name__ == "__main__":
sys.exit(main())