-
Notifications
You must be signed in to change notification settings - Fork 5
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Adds two modular constraints necessary for a network flow formulation, useful for example for single particle tracking: - Symmetry: # incoming edges = # outgoing edges - Minimum track length: Each starting track needs to contain at least the given number of edges. Currently only implemented for length 1, i.e. each node with a set appear indicator needs to have at least one outgoing edge. Each constraint includes a functional test.
- Loading branch information
1 parent
dc797d4
commit f125d21
Showing
4 changed files
with
161 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,15 +1,19 @@ | ||
from .constraint import Constraint | ||
from .expression import ExpressionConstraint | ||
from .in_out_symmetry import InOutSymmetry | ||
from .max_children import MaxChildren | ||
from .max_parents import MaxParents | ||
from .min_track_length import MinTrackLength | ||
from .pin import Pin | ||
from .select_edge_nodes import SelectEdgeNodes | ||
|
||
__all__ = [ | ||
"Constraint", | ||
"ExpressionConstraint", | ||
"InOutSymmetry", | ||
"MaxChildren", | ||
"MaxParents", | ||
"MinTrackLength", | ||
"Pin", | ||
"SelectEdgeNodes", | ||
] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,50 @@ | ||
from __future__ import annotations | ||
|
||
from typing import TYPE_CHECKING | ||
|
||
import ilpy | ||
|
||
from motile.constraints import Constraint | ||
from motile.variables import EdgeSelected | ||
|
||
if TYPE_CHECKING: | ||
from motile.solver import Solver | ||
|
||
|
||
class InOutSymmetry(Constraint): | ||
r"""Ensures that all nodes, apart from the ones in the first and last | ||
frame, have as many incoming edges as outgoing edges. | ||
Adds the following linear constraint for nodes :math:`v` not in first or | ||
last frame: | ||
.. math:: | ||
\sum_{e \in \\text{in_edges}(v)} x_e = \sum{e \in \\text{out_edges}(v)} x_e | ||
""" | ||
|
||
def instantiate(self, solver: Solver) -> list[ilpy.Constraint]: | ||
edge_indicators = solver.get_variables(EdgeSelected) | ||
start, end = solver.graph.get_frames() | ||
|
||
constraints = [] | ||
for node, attrs in solver.graph.nodes.items(): | ||
constraint = ilpy.Constraint() | ||
|
||
if solver.graph.frame_attribute in attrs and attrs[ | ||
solver.graph.frame_attribute | ||
] not in ( | ||
start, | ||
end - 1, # type: ignore | ||
): | ||
for prev_edge in solver.graph.prev_edges[node]: | ||
ind_e = edge_indicators[prev_edge] | ||
constraint.set_coefficient(ind_e, 1) | ||
for next_edge in solver.graph.next_edges[node]: | ||
ind_e = edge_indicators[next_edge] | ||
constraint.set_coefficient(ind_e, -1) | ||
constraint.set_relation(ilpy.Relation.Equal) | ||
constraint.set_value(0) | ||
|
||
constraints.append(constraint) | ||
|
||
return constraints |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,42 @@ | ||
from __future__ import annotations | ||
|
||
from typing import TYPE_CHECKING | ||
|
||
import ilpy | ||
|
||
from motile.constraints import Constraint | ||
from motile.variables import EdgeSelected, NodeAppear | ||
|
||
if TYPE_CHECKING: | ||
pass | ||
|
||
|
||
class MinTrackLength(Constraint): | ||
r"""Ensures that each appearing track consists of at least ``min_edges`` | ||
edges. | ||
Currently only supports ``min_edges = 1``. | ||
Args: | ||
min_edges: The minimum number of edges per track. | ||
""" | ||
|
||
def __init__(self, min_edges: int) -> None: | ||
if min_edges != 1: | ||
raise NotImplementedError( | ||
"Can only enforce minimum track length of 1 edge." | ||
) | ||
self.min_edges = min_edges | ||
|
||
def instantiate(self, solver): | ||
appear_indicators = solver.get_variables(NodeAppear) | ||
edge_indicators = solver.get_variables(EdgeSelected) | ||
for node in solver.graph.nodes: | ||
constraint = ilpy.Constraint() | ||
constraint.set_coefficient(appear_indicators[node], 1) | ||
for edge in solver.graph.next_edges[node]: | ||
constraint.set_coefficient(edge_indicators[edge], -1) | ||
constraint.set_relation(ilpy.Relation.LessEqual) | ||
constraint.set_value(0) | ||
yield constraint |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters