Skip to content

Commit

Permalink
Epic backtracking constraint
Browse files Browse the repository at this point in the history
  • Loading branch information
Kris Brown committed May 24, 2024
1 parent 3ddc986 commit 5a31b28
Show file tree
Hide file tree
Showing 2 changed files with 53 additions and 4 deletions.
39 changes: 35 additions & 4 deletions src/categorical_algebra/HomSearch.jl
Original file line number Diff line number Diff line change
Expand Up @@ -177,7 +177,7 @@ been bound.
struct BacktrackingState{
Dom <: ACSet, Codom <: ACSet,
Assign <: NamedTuple, PartialAssign <: NamedTuple, LooseFun <: NamedTuple,
Predicates <: NamedTuple
Predicates <: NamedTuple,Image <: NamedTuple, Unassign<: NamedTuple
}
assignment::Assign
assignment_depth::Assign
Expand All @@ -186,10 +186,13 @@ struct BacktrackingState{
codom::Codom
type_components::LooseFun
predicates::Predicates
image::Image # Negative of image for epic components or if finding an epimorphism
unassigned::Unassign # "# of unassigned elems in domain of a component

end

function backtracking_search(f, X::ACSet, Y::ACSet;
monic=false, iso=false, random=false, predicates=(;),
monic=false, epic=false, iso=false, random=false, predicates=(;),
type_components=(;), initial=(;), error_failures=false)
S, Sy = acset_schema.([X,Y])
S == Sy || error("Schemas must match for morphism search")
Expand All @@ -212,15 +215,21 @@ function backtracking_search(f, X::ACSet, Y::ACSet;
if monic isa Bool
monic = monic ? ObAttr : ()
end
if epic isa Bool
epic = epic ? Ob : ()
end
iso_failures = Iterators.filter(c->nparts(X,c)!=nparts(Y,c),iso)
mono_failures = Iterators.filter(c->nparts(X,c)>nparts(Y,c),monic)
if (!isempty(iso_failures) || !isempty(mono_failures))
epi_failures = Iterators.filter(c->nparts(X,c)<nparts(Y,c),epic)

if !all(isempty, [iso_failures,mono_failures,epi_failures])
if !error_failures
return false
else error("""
Cardinalities inconsistent with request for...
iso at object(s) $iso_failures
mono at object(s) $mono_failures
epi at object(s) $epi_failures
""")
end
end
Expand Down Expand Up @@ -249,8 +258,15 @@ function backtracking_search(f, X::ACSet, Y::ACSet;
(c in monic ? zeros(Int, maxpart(Y, c)) : nothing) for c in ObAttr)
loosefuns = NamedTuple{Attr}(
isnothing(type_components) ? identity : get(type_components, c, identity) for c in Attr)

images = NamedTuple{Ob}(
(c in epic ? zeros(Int, nparts(Y, c)) : nothing) for c in Ob)
unassigned = NamedTuple{Ob}(
(c in epic ? [nparts(X, c)] : nothing) for c in Ob)

state = BacktrackingState(assignment, assignment_depth,
inv_assignment, X, Y, loosefuns, pred_nt)
inv_assignment, X, Y, loosefuns, pred_nt,
images, unassigned)

# Make any initial assignments, failing immediately if inconsistent.
for (c, c_assignments) in pairs(initial)
Expand Down Expand Up @@ -350,6 +366,13 @@ assign_elem!(state::BacktrackingState{<:DynamicACSet}, depth, c, x, y) =
return false
end

# With an epic constraint, fail based on the # of unassigned in dom vs codom
if (!isnothing(state.image[@ct c]) && state.image[@ct c][y]!=0
&& only(state.unassigned[@ct c]) <= count(==(0), state.image[@ct c]))
return false
end


isnothing(state.predicates[c][x]) || y state.predicates[c][x] || return false

# Check attributes first to fail as quickly as possible.
Expand All @@ -373,6 +396,10 @@ assign_elem!(state::BacktrackingState{<:DynamicACSet}, depth, c, x, y) =
if !isnothing(state.inv_assignment[@ct c])
state.inv_assignment[@ct c][y] = x
end
if !isnothing(state.image[@ct c])
state.image[@ct c][y] += 1
state.unassigned[@ct c][1] -= 1
end

@ct_ctrl for (f,_,d) in attrs(S; from=c)
if subpart(X,x,@ct(f)) isa AttrVar
Expand Down Expand Up @@ -405,6 +432,10 @@ unassign_elem!(state::BacktrackingState{<:DynamicACSet}, depth, c, x) =
y = state.assignment[@ct c][x]
state.inv_assignment[@ct c][y] = 0
end
if !isnothing(state.unassigned[@ct c])
state.unassigned[@ct c][1] += 1
state.image[@ct c][state.assignment[@ct c][x]] -= 1
end

@ct_ctrl for (f,_,d) in attrs(S; from=c)
if subpart(X,x,@ct(f)) isa AttrVar
Expand Down
18 changes: 18 additions & 0 deletions test/categorical_algebra/HomSearch.jl
Original file line number Diff line number Diff line change
Expand Up @@ -77,6 +77,24 @@ g1, g2 = path_graph(Graph, 3), path_graph(Graph, 2)
rem_part!(g1,:E,2)
@test_throws ErrorException homomorphism(g1,g2;monic=true,error_failures=true)

# Epic constraint
g0, g1, g2 = Graph(2), Graph(2), Graph(2)
add_edges!(g0, [1,1],[1,2]) # ↻•→•
add_edges!(g1, [1,1],[2,2]) # •⇉•
add_edges!(g2, [1,1,1],[1,1,2]) # ↻↻•→•
@test length(homomorphisms(g1, g2, epic=[:V])) == 1
@test length(homomorphisms(g1, g2, epic=[:E])) == 0
@test length(homomorphisms(g2, g0, epic=[:E])) == 1
@test length(homomorphisms(g2, g0, epic=[:V])) == 1

g3, g4 = path_graph(Graph,3), path_graph(Graph,4)
add_edges!(g3,[1,3],[1,3]) # g3: ↻•→•→• ↺
@test length(homomorphisms(g4,g3)) == 6 # 2 for each: 1/2/3 edges sent to loop
@test length(homomorphisms(g4,g3; epic=[:V])) == 2 # send only one edge to loop
@test length(homomorphisms(g4,g3; epic=[:E])) == 0 # only have 3 edges to map

@test length(homomorphisms(Graph(4),Graph(2); epic=true)) == 14 # 2^4 - 2

# Symmetric graphs
#-----------------

Expand Down

0 comments on commit 5a31b28

Please sign in to comment.