diff --git a/.github/workflows/stale.yml b/.github/workflows/stale.yml new file mode 100644 index 000000000..73551256f --- /dev/null +++ b/.github/workflows/stale.yml @@ -0,0 +1,28 @@ +# This workflow warns and then closes issues and PRs that have had no activity for a specified amount of time. +# +# You can adjust the behavior by modifying this file. +# For more information, see: +# https://github.com/actions/stale +name: Mark stale issues and pull requests + +on: + schedule: + - cron: '42 0 * * *' + +jobs: + stale: + + runs-on: ubuntu-latest + permissions: + issues: write + pull-requests: write + + steps: + - uses: actions/stale@v3 + with: + repo-token: ${{ secrets.GITHUB_TOKEN }} + stale-issue-message: 'Stale issue message' + stale-pr-message: 'Stale pull request message' + stale-issue-label: 'no-issue-activity' + stale-pr-label: 'no-pr-activity' + days-before-close: -1 diff --git a/doc/attr.xml b/doc/attr.xml index 6c1d130cd..f2862fc47 100644 --- a/doc/attr.xml +++ b/doc/attr.xml @@ -2430,3 +2430,40 @@ gap> Length(M); <#/GAPDoc> + +<#GAPDoc Label="NonUpperSemimodularPair"> + + + + A pair of vertices or fail. + + NonUpperSemimodularPair returns a pair of vertices in the digraph + D that witnesses the fact that D does not represent an upper + semimodular lattice, if such a pair exists.

+ + If the digraph D does not satisfy , then an error is given. Otherwise if the + digraph D does satisfy , then either a non-upper semimodular pair of + vertices is returned, or fail is returned if no such pair exists + (meaning that D is an upper semimodular lattice.

+ + NonLowerSemimodularPair behaves in the analogous way to + NonUpperSemimodularPair with respect to lower semimodularity.

+ + See and for the definition of upper + semimodularity of a lattice. + + D := DigraphFromDigraph6String( +> "&M~~sc`lYUZO__KIBboC_@h?U_?_GL?A_?c"); + +gap> NonLowerSemimodularPair(D); +[ 10, 9 ] +gap> NonUpperSemimodularPair(D); +fail +]]> + + +<#/GAPDoc> diff --git a/doc/digraph.xml b/doc/digraph.xml index ea43bb730..9f1e0159f 100644 --- a/doc/digraph.xml +++ b/doc/digraph.xml @@ -421,33 +421,52 @@ gap> D := DigraphByInNeighbours(IsMutableDigraph, <#GAPDoc Label="AsDigraph"> - + A digraph, or fail. - If f is a transformation or permutation, - and n is a non-negative integer - such that the restriction of f to [1..n] defines - a function of [1..n], then AsDigraph - returns the functional digraph with n vertices defined by - f. See . -

- - Specifically, the digraph returned by AsDigraph has n edges: - for each vertex v in [1..n], there is a unique edge - with source v; this edge has range v^f. -

- If the optional second argument n is not supplied, then - the degree of the transformation f, - or the largest moved point of the permutation f, - as applicable, is used by default. If the restriction of - f to [1..n] does not define a function of - [1..n], then AsDigraph(f, n) - returns fail. + If f is a binary relation represented as one of the following in + &GAP;: + + + a transformation + + + satisfying ; + + + a permutation + + + satisfying ; + + + a partial perm + + + satisfying ; + + + a binary relation + + + satisfying ; + + + and n is a non-negative integer, then AsDigraph attempts + to construct a digraph with n vertices whose edges are determined + by f.

+ + The digraph returned by AsDigraph has for each vertex + v in [1 .. n], an edge with source v and range + v ^ f. If v ^ f is greater than n for any + v, then fail is returned.

- See also . + If the optional second argument n is not supplied, then the degree + of the transformation f, the largest moved point of the permutation + f, the maximum of the degree and the codegree of the partial perm + f, or as applicable, is used by default.

&STANDARD_FILT_TEXT; diff --git a/doc/prop.xml b/doc/prop.xml index e8b720471..ee75b409b 100644 --- a/doc/prop.xml +++ b/doc/prop.xml @@ -1459,3 +1459,39 @@ true <#/GAPDoc> + +<#GAPDoc Label="IsUpperSemimodularDigraph"> + + + + true or false. + + IsUpperSemimodularDigraph returns true if the digraph + D represents an upper semimodular lattice and false if it + does not. Similarly, IsLowerSemimodularDigraph returns true + if D represents a lower semimodular lattice and false if + it does not.

+ + In a lattice we say that a vertex a covers a vertex b + if a is greater than b, and there are no further vertices + between a and b. A lattice is upper semimodular if + whenever the meet of a and b is covered by a, the join + of a and b covers b. Lower semimodularity is + defined analogously.

+ + See also , , + and . + + &MUTABLE_RECOMPUTED_PROP; + + D := DigraphFromDigraph6String( +> "&M~~sc`lYUZO__KIBboC_@h?U_?_GL?A_?c"); + +gap> IsUpperSemimodularDigraph(D); +true +gap> IsLowerSemimodularDigraph(D); +false]]> + + +<#/GAPDoc> diff --git a/doc/z-chap4.xml b/doc/z-chap4.xml index 00224832d..1524d671c 100644 --- a/doc/z-chap4.xml +++ b/doc/z-chap4.xml @@ -35,10 +35,14 @@ <#Include Label="InDegreeOfVertex"> <#Include Label="InNeighboursOfVertex"> <#Include Label="DigraphLoops"> - <#Include Label="PartialOrderDigraphMeetOfVertices"> <#Include Label="DegreeMatrix"> <#Include Label="LaplacianMatrix"> + +

Orders + <#Include Label="PartialOrderDigraphMeetOfVertices"> + <#Include Label="NonUpperSemimodularPair"> +
Reachability and connectivity <#Include Label="DigraphDiameter"> diff --git a/doc/z-chap5.xml b/doc/z-chap5.xml index 906c65d76..f571974c2 100644 --- a/doc/z-chap5.xml +++ b/doc/z-chap5.xml @@ -22,9 +22,13 @@ <#Include Label="IsSymmetricDigraph"> <#Include Label="IsTournament"> <#Include Label="IsTransitiveDigraph"> +
+ +
Orders <#Include Label="IsPreorderDigraph"> <#Include Label="IsPartialOrderDigraph"> <#Include Label="IsMeetSemilatticeDigraph"> + <#Include Label="IsUpperSemimodularDigraph">
Regularity diff --git a/gap/attr.gd b/gap/attr.gd index 5336d5b75..b67221290 100644 --- a/gap/attr.gd +++ b/gap/attr.gd @@ -121,3 +121,9 @@ DeclareAttribute("DigraphMaximumMatching", IsDigraph); DeclareAttribute("Bridges", IsDigraph); DeclareAttributeThatReturnsDigraph("StrongOrientation", IsDigraph); + +DeclareAttribute("NonUpperSemimodularPair", IsDigraph); +DeclareAttribute("NonLowerSemimodularPair", IsDigraph); + +DeclareProperty("IsUpperSemimodularDigraph", IsDigraph); +DeclareProperty("IsLowerSemimodularDigraph", IsDigraph); diff --git a/gap/attr.gi b/gap/attr.gi index 01a5e68f7..045e842b8 100644 --- a/gap/attr.gi +++ b/gap/attr.gi @@ -2655,3 +2655,73 @@ function(D) M := List(DigraphLoops(D), x -> [x, x]); return Union(M, DIGRAPHS_MateToMatching(D, mateD)); end); + +# The following function is a transliteration from python to GAP of +# the function find_nonsemimodular_pair +# in sage/src/sage/combinat/posets/hasse_diagram.py + +BindGlobal("DIGRAPHS_NonSemimodularPair", +function(nbs) + local n, covers, covers_len, a, covers_a, b, e, a_i, b_i; + n := Length(nbs); + + for e in [1 .. n] do + covers := nbs[e]; + covers_len := Length(covers); + if covers_len < 2 then + continue; + fi; + for a_i in [1 .. covers_len] do + a := covers[a_i]; + covers_a := nbs[a]; + for b_i in [1 .. a_i] do + b := covers[b_i]; + if not ForAny(nbs[b], j -> j in covers_a) then + return [a, b]; + fi; + od; + od; + od; + + return fail; +end); + +InstallMethod(NonUpperSemimodularPair, "for a digraph", +[IsDigraphByOutNeighboursRep], +function(D) + if not IsLatticeDigraph(D) then + ErrorNoReturn("the argument (a digraph) is not a lattice"); + fi; + D := DigraphReflexiveTransitiveReduction(DigraphMutableCopyIfMutable(D)); + return DIGRAPHS_NonSemimodularPair(OutNeighbours(D)); +end); + +InstallMethod(NonLowerSemimodularPair, "for a digraph", +[IsDigraphByOutNeighboursRep], +function(D) + if not IsLatticeDigraph(D) then + ErrorNoReturn("the argument (a digraph) is not a lattice"); + fi; + D := DigraphReflexiveTransitiveReduction(DigraphMutableCopyIfMutable(D)); + return DIGRAPHS_NonSemimodularPair(InNeighbours(D)); +end); + +InstallMethod(IsUpperSemimodularDigraph, "for a digraph", +[IsDigraphByOutNeighboursRep], +function(D) + if not IsLatticeDigraph(D) then + return false; + fi; + D := DigraphReflexiveTransitiveReduction(DigraphMutableCopyIfMutable(D)); + return DIGRAPHS_NonSemimodularPair(OutNeighbours(D)) = fail; +end); + +InstallMethod(IsLowerSemimodularDigraph, "for a digraph", +[IsDigraphByOutNeighboursRep], +function(D) + if not IsLatticeDigraph(D) then + return false; + fi; + D := DigraphReflexiveTransitiveReduction(DigraphMutableCopyIfMutable(D)); + return DIGRAPHS_NonSemimodularPair(InNeighbours(D)) = fail; +end); diff --git a/gap/digraph.gd b/gap/digraph.gd index d93389b46..0f722b5a3 100644 --- a/gap/digraph.gd +++ b/gap/digraph.gd @@ -106,18 +106,24 @@ DeclareSynonym("DigraphByInNeighbors", DigraphByInNeighbours); DeclareConstructor("AsDigraphCons", [IsDigraph, IsBinaryRelation]); DeclareConstructor("AsDigraphCons", [IsDigraph, IsTransformation]); DeclareConstructor("AsDigraphCons", [IsDigraph, IsTransformation, IsInt]); +DeclareConstructor("AsDigraphCons", [IsDigraph, IsPartialPerm]); +DeclareConstructor("AsDigraphCons", [IsDigraph, IsPartialPerm, IsInt]); DeclareOperation("AsDigraph", [IsFunction, IsBinaryRelation]); DeclareOperation("AsDigraph", [IsFunction, IsTransformation]); DeclareOperation("AsDigraph", [IsFunction, IsTransformation, IsInt]); DeclareOperation("AsDigraph", [IsFunction, IsPerm]); DeclareOperation("AsDigraph", [IsFunction, IsPerm, IsInt]); +DeclareOperation("AsDigraph", [IsFunction, IsPartialPerm]); +DeclareOperation("AsDigraph", [IsFunction, IsPartialPerm, IsInt]); DeclareOperation("AsDigraph", [IsBinaryRelation]); DeclareOperation("AsDigraph", [IsTransformation]); DeclareOperation("AsDigraph", [IsTransformation, IsInt]); DeclareOperation("AsDigraph", [IsPerm]); DeclareOperation("AsDigraph", [IsPerm, IsInt]); +DeclareOperation("AsDigraph", [IsPartialPerm]); +DeclareOperation("AsDigraph", [IsPartialPerm, IsInt]); DeclareOperation("AsBinaryRelation", [IsDigraph]); DeclareOperation("AsSemigroup", [IsFunction, IsDigraph]); diff --git a/gap/digraph.gi b/gap/digraph.gi index e400ff2e9..c0ef7a829 100644 --- a/gap/digraph.gi +++ b/gap/digraph.gi @@ -1062,6 +1062,58 @@ InstallMethod(AsDigraph, "for a function and a perm", InstallMethod(AsDigraph, "for a perm", [IsPerm], p -> AsDigraph(AsTransformation(p))); +InstallMethod(AsDigraphCons, +"for IsMutableDigraph, a partial perm, and an integer", +[IsMutableDigraph, IsPartialPerm, IsInt], +function(filt, f, n) + local list, x, i; + if n < 0 then + ErrorNoReturn("the 2nd argument should be a non-negative integer,"); + fi; + + list := EmptyPlist(n); + for i in [1 .. n] do + x := i ^ f; + if x > n then + return fail; + elif x <> 0 then + list[i] := [x]; + else + list[i] := []; + fi; + od; + return DigraphNC(IsMutableDigraph, list); +end); + +InstallMethod(AsDigraphCons, +"for IsImmutableDigraph, a partial perm, and an integer", +[IsImmutableDigraph, IsPartialPerm, IsInt], +function(filt, f, n) + local D; + D := AsDigraph(IsMutableDigraph, f, n); + if D <> fail then + D := MakeImmutable(D); + SetIsMultiDigraph(D, false); + fi; + return D; +end); + +InstallMethod(AsDigraph, "for a function, a partial perm, and an integer", +[IsFunction, IsPartialPerm, IsInt], AsDigraphCons); + +InstallMethod(AsDigraph, "for a partial perm and an integer", +[IsPartialPerm, IsInt], +{t, n} -> AsDigraphCons(IsImmutableDigraph, t, n)); + +InstallMethod(AsDigraph, "for a function and a partial perm", +[IsFunction, IsPartialPerm], +{func, t} -> AsDigraphCons(func, t, Maximum(DegreeOfPartialPerm(t), + CodegreeOfPartialPerm(t)))); + +InstallMethod(AsDigraph, "for a partial perm", [IsPartialPerm], +t -> AsDigraphCons(IsImmutableDigraph, t, Maximum(DegreeOfPartialPerm(t), + CodegreeOfPartialPerm(t)))); + InstallMethod(AsBinaryRelation, "for a digraph", [IsDigraphByOutNeighboursRep], function(D) local rel; diff --git a/tst/standard/attr.tst b/tst/standard/attr.tst index 9f2d76de9..5c0558caf 100644 --- a/tst/standard/attr.tst +++ b/tst/standard/attr.tst @@ -2792,6 +2792,36 @@ gap> D := DigraphRemoveEdge(D, 1, 3); gap> D := DigraphRemoveEdge(D, 1, 3); +# Semimodular lattices +gap> D := DigraphFromDigraph6String("&C[o?"); + +gap> IsUpperSemimodularDigraph(D); +false +gap> IsLowerSemimodularDigraph(D); +false +gap> NonUpperSemimodularPair(D); +Error, the argument (a digraph) is not a lattice +gap> NonLowerSemimodularPair(D); +Error, the argument (a digraph) is not a lattice +gap> D := DigraphFromDigraph6String("&K~~]mKaC_EgLb?_?~?g?m?a?b"); + +gap> IsUpperSemimodularDigraph(D); +true +gap> NonUpperSemimodularPair(D); +fail +gap> IsLowerSemimodularDigraph(D); +true +gap> NonLowerSemimodularPair(D); +fail +gap> D := DigraphFromDigraph6String("&M~~sc`lYUZO__KIBboC_@h?U_?_GL?A_?c"); + +gap> IsUpperSemimodularDigraph(D); +true +gap> IsLowerSemimodularDigraph(D); +false +gap> NonLowerSemimodularPair(D); +[ 10, 9 ] + # DIGRAPHS_UnbindVariables gap> Unbind(adj); gap> Unbind(adj1); diff --git a/tst/standard/digraph.tst b/tst/standard/digraph.tst index 3ad78e781..d8cbffcb6 100644 --- a/tst/standard/digraph.tst +++ b/tst/standard/digraph.tst @@ -479,7 +479,8 @@ fail gap> f := ();; gap> D := AsDigraph(f); -gap> D = EmptyDigraph(0);; +gap> D = EmptyDigraph(0); +true gap> AsDigraph(f, 10); gap> g := (1, 3, 7)(2, 6, 5, 8);; @@ -511,6 +512,41 @@ gap> D := AsDigraph(IsMutableDigraph, h, 6); gap> OutNeighbours(D); [ [ 1 ], [ 5 ], [ 2 ], [ 4 ], [ 3 ], [ 6 ] ] +# AsDigraph for a partial perm +gap> f := PartialPerm([]);; +gap> D := AsDigraph(f); + +gap> D = EmptyDigraph(0); +true +gap> AsDigraph(f, 10); + +gap> x := AsPartialPerm((1, 3, 7)(2, 6, 5, 8)); +(1,3,7)(2,6,5,8)(4) +gap> D := AsDigraph(x); + +gap> AsDigraph(x, -1); +Error, the 2nd argument should be a non-negative integer, +gap> AsDigraph(x, 0); + +gap> D := AsDigraph(g, 10); + +gap> AsDigraph(g, 7); +fail +gap> x := AsPartialPerm((2, 5, 3), 5); +(1)(2,5,3)(4) +gap> D := AsDigraph(IsMutableDigraph, x); + +gap> AsDigraph(IsImmutableDigraph, x, 5); + +gap> D = AsDigraph(IsImmutableDigraph, x, 5); +true +gap> D := AsDigraph(IsMutableDigraph, x, 6); + +gap> OutNeighbours(D); +[ [ 1 ], [ 5 ], [ 2 ], [ 4 ], [ 3 ], [ ] ] +gap> AsDigraph(AsPartialPerm((2, 5, 3)), 2); +fail + # RandomDigraph gap> IsImmutableDigraph(RandomDigraph(100, 0.2)); true