routing-subtour-elimination
$
npx mdskill add elizaOS/eliza/routing-subtour-eliminationIn routing MIPs, degree and continuity constraints are not enough. A vehicle can have one depot-to-depot path and a separate closed cycle among stations. Add subtour-elimination constraints whenever binary arc variables decide routes.
SKILL.md
.github/skills/routing-subtour-eliminationView on GitHub ↗
---
name: routing-subtour-elimination
description: Subtour-elimination methods for TSP, VRP, pickup/dropoff routing, and routing MIPs with binary arc variables. Use when route-continuity constraints may permit disconnected cycles and the model needs MTZ constraints, flow-based connectivity constraints, DFJ subset cuts, or lazy/iterative subtour cuts.
---
# Routing Subtour Elimination
In routing MIPs, degree and continuity constraints are not enough. A vehicle can have one depot-to-depot path and a separate closed cycle among stations. Add subtour-elimination constraints whenever binary arc variables decide routes.
Use this base notation:
```python
START = "depot_start"
END = "depot_end"
vehicles = range(K)
stations = range(n)
from_nodes = [START, *stations]
to_nodes = [*stations, END]
arcs = [(i, j) for i in from_nodes for j in to_nodes if i != j and not (i == START and j == END)]
x = {(v, i, j): model.addVar(vtype="B", name=f"x_{v}_{i}_{j}") for v in vehicles for i, j in arcs}
```
## Required Base Route Constraints
Subtour elimination assumes each selected station has matching inbound and outbound route arcs.
```python
for v in vehicles:
model.addCons(quicksum(x[v, START, j] for j in stations) == 1)
model.addCons(quicksum(x[v, i, END] for i in stations) == 1)
for i in stations:
incoming = quicksum(x[v, j, i] for j in from_nodes if j != i)
outgoing = quicksum(x[v, i, j] for j in to_nodes if j != i)
model.addCons(incoming == outgoing)
model.addCons(outgoing <= 1)
```
The subtour methods below prevent station-only cycles that are disconnected from `START`.
## 1. MTZ Order Constraints
MTZ adds an order variable for each vehicle-station pair. If vehicle `v` travels from station `i` to station `j`, then `order[v, j]` must be greater than `order[v, i]`.
```python
order = {
(v, i): model.addVar(vtype="C", lb=1, ub=max(1, n), name=f"order_{v}_{i}")
for v in vehicles
for i in stations
}
for v in vehicles:
for i in stations:
for j in stations:
if i != j:
model.addCons(order[v, i] - order[v, j] + n * x[v, i, j] <= n - 1)
```
Pros:
- Compact: `O(K n^2)` constraints and `O(K n)` extra variables.
- Easy to implement in common Python optimization APIs.
- Good default for small and medium benchmark instances.
Cons:
- LP relaxation is weak compared with cutset or flow formulations.
- Can be slow for larger VRPs.
- Order variables are artificial; do not interpret them as service times unless you also model time.
Use MTZ first when correctness and implementation speed matter more than best possible MIP strength.
## 2. Single-Commodity Flow Connectivity
Add an artificial connectivity flow that starts at the depot and sends one unit to every visited station. This flow is not physical vehicle load.
```python
visit = {
(v, i): quicksum(x[v, i, j] for j in to_nodes if j != i)
for v in vehicles
for i in stations
}
flow_arcs = [(i, j) for i in [START, *stations] for j in stations if i != j]
f = {
(v, i, j): model.addVar(vtype="C", lb=0, ub=n, name=f"conn_flow_{v}_{i}_{j}")
for v in vehicles
for i, j in flow_arcs
}
for v in vehicles:
total_visits = quicksum(visit[v, i] for i in stations)
model.addCons(quicksum(f[v, START, j] for j in stations) == total_visits)
for i, j in flow_arcs:
model.addCons(f[v, i, j] <= n * x[v, i, j])
for i in stations:
incoming_flow = quicksum(f[v, h, i] for h in [START, *stations] if h != i)
outgoing_flow = quicksum(f[v, i, j] for j in stations if j != i)
model.addCons(incoming_flow - outgoing_flow == visit[v, i])
```
Pros:
- Stronger connectivity logic than MTZ in many models.
- Static constraints, no callback needed.
- Works with optional station visits.
Cons:
- Adds `O(K n^2)` continuous variables.
- Do not reuse truck load as the connectivity flow when the vehicle can both pick up and drop off. Physical load can increase and decrease; connectivity flow should monotonically distribute artificial units.
- More memory than MTZ.
Use this when MTZ gives weak incumbents or slow progress and the instance is still modest in size.
## 3. DFJ Subset Cuts
For every nonempty proper subset `S` of stations, selected station-to-station arcs inside `S` cannot form a closed cycle:
```text
sum_{i in S, j in S, i != j} x[v,i,j] <= |S| - 1
```
For very small `n`, static enumeration is possible:
```python
from itertools import combinations
for v in vehicles:
for r in range(2, n):
for S_tuple in combinations(stations, r):
S = set(S_tuple)
model.addCons(
quicksum(x[v, i, j] for i in S for j in S if i != j) <= len(S) - 1
)
```
Pros:
- Strong, direct subtour elimination.
- No artificial order or flow variables.
Cons:
- Exponential number of constraints.
- Static enumeration is only acceptable for small station counts.
Use static DFJ only for tiny instances or as a debugging baseline.
## 4. Lazy or Iterative Cut Separation
The strongest practical pattern is to solve with base route constraints, detect subtours in incumbents, add only the violated DFJ cuts, and continue.
In solvers with convenient lazy callbacks, add cuts during branch-and-bound. Some Python solver APIs require callback or constraint-handler plumbing for true lazy enforcement, so iterative cut separation is often simpler for portable benchmark code:
```python
def selected_arcs(model, x, v, arcs):
return [(i, j) for i, j in arcs if model.getVal(x[v, i, j]) > 0.5]
def station_cycles_without_start(selected, stations):
succ = {i: j for i, j in selected}
cycles = []
seen = set()
for start in stations:
if start in seen or start not in succ:
continue
path = []
cur = start
pos = {}
while cur in succ and cur not in pos and cur not in seen:
pos[cur] = len(path)
path.append(cur)
cur = succ[cur]
seen.update(path)
if cur in pos:
cycle = path[pos[cur]:]
if START not in cycle and END not in cycle:
cycles.append(cycle)
return cycles
while True:
model.optimize()
if model.getNSols() == 0:
raise RuntimeError(f"no feasible solution; status={model.getStatus()}")
cuts_added = 0
for v in vehicles:
selected = selected_arcs(model, x, v, arcs)
for cycle in station_cycles_without_start(selected, stations):
if len(cycle) >= 2:
S = set(cycle)
model.freeTransform()
model.addCons(
quicksum(x[v, i, j] for i in S for j in S if i != j) <= len(S) - 1
)
cuts_added += 1
if cuts_added == 0:
break
```
Pros:
- Adds only cuts that are needed.
- Often stronger than MTZ.
- Avoids exponential static SEC generation.
Cons:
- Iterative resolve can be slower than a true callback.
- Requires reliable subtour detection.
- More moving parts than MTZ.
Use this when static MTZ is too weak and the solver environment does not make lazy callbacks convenient.
## Method Choice
| Method | Best For | Avoid When |
| --- | --- | --- |
| MTZ | Quick, compact, small/medium MIPs | Large hard VRPs where relaxation strength matters |
| Single-commodity flow | Stronger static connectivity, optional visits | Memory is tight, or `n` is large |
| Multi-commodity flow | Very strong small routing models | Most practical benchmark tasks; too many variables |
| Static DFJ | Tiny instances, debugging | More than roughly 15-18 stations without careful filtering |
| Lazy/iterative DFJ cuts | Strong routing models with many possible SECs | Solver API/callback complexity is too risky |
For pickup/dropoff rebalancing, start with MTZ or artificial connectivity flow. Do not use physical truck load as the only subtour-elimination mechanism because pickup/dropoff load can increase and decrease and may not prove route connectivity.
## Validation
After solving, reconstruct each route by following selected arcs:
```python
def extract_route(selected):
outgoing = dict(selected)
route = [START]
cur = START
seen = {START}
while cur != END:
if cur not in outgoing:
raise RuntimeError(f"route disconnected at {cur!r}")
cur = outgoing[cur]
if cur in seen and cur != END:
raise RuntimeError(f"cycle detected at {cur!r}")
route.append(cur)
seen.add(cur)
return route
```
Fail fast if a selected solution has a disconnected cycle, repeated station, missing depot start, or missing depot end.
More from elizaOS/eliza
- ac-branch-pi-modelAC branch pi-model power flow equations (P/Q and |S|) with transformer tap ratio and phase shift, matching `acopf-math-model.md` and MATPOWER branch fields. Use when computing branch flows in either direction, aggregating bus injections for nodal balance, checking MVA (rateA) limits, computing branch loading %, or debugging sign/units issues in AC power flow.
- academic-pdf-redactionRedact text from PDF documents for blind review anonymization
- ada-plan-view-accessibilityUse when checking simplified ADA-derived plan-view bathroom accessibility constraints such as turning space, door clear width, toilet centerline, grab bars, and lavatory knee/toe clearance.
- analyze-ciAnalyze failed GitHub Action jobs for a pull request.
- architectural-dxf-extractionUse when extracting plan-view architectural geometry from DXF files with semantic CAD layers, especially when outputs must normalize rooms, doors, fixtures, clearances, and grab bars into machine-checkable JSON.
- attitude-controller-plannerUse this skill when implementing the inner control loop for a quadrotor — attitude (roll/pitch/yaw) PID control and attitude planning (converting desired acceleration to desired Euler angles). Covers gain layout, integral reset pattern, and the attitude planner inverse kinematics.
- azure-bgpAnalyze and resolve BGP oscillation and BGP route leaks in Azure Virtual WAN–style hub-and-spoke topologies (and similar cloud-managed BGP environments). Detect preference cycles, identify valley-free violations, and propose allowed policy-level mitigations while rejecting prohibited fixes.
- box-least-squaresBox Least Squares (BLS) periodogram for detecting transiting exoplanets and eclipsing binaries. Use when searching for periodic box-shaped dips in light curves. Alternative to Transit Least Squares, available in astropy.timeseries. Based on Kovács et al. (2002).
- browser-testingVERIFY your changes work. Measure CLS, detect theme flicker, test visual stability, check performance. Use BEFORE and AFTER making changes to confirm fixes. Includes ready-to-run scripts: measure-cls.ts, detect-flicker.ts
- cache-policy-comparisonCompare and implement eviction policies (LRU, LFU, FIFO, S3FIFO, ARC) for bounded-capacity caches. Use when choosing or implementing an eviction policy for a buffer pool, page cache, CDN edge, or LLM KV cache, or when writing a replay simulator that supports multiple policies. Clarifies recency vs frequency semantics, queue topology, saturating counters, ghost buffers, and the second-chance rule that distinguishes modern FIFO-family policies from classic LRU.