Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add constructor for Dicyclic groups and alias for Quaternion groups #4661

Open
wants to merge 11 commits into
base: master
Choose a base branch
from

Conversation

jamesnohilly
Copy link
Collaborator

@jamesnohilly jamesnohilly commented Feb 27, 2025

This PR adds a group constructor for Dicyclic Groups (dicyclic_group) and changes the constructor for Quaternion Groups to be an alias of dicyclic_group.

This PR closes #1630

@@ -827,6 +827,8 @@ as an instance of `T`,
where `n` is a power of 2 and `T` is in
Copy link
Collaborator

@thofma thofma Feb 27, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
where `n` is a power of 2 and `T` is in
where `n` is a multiple of 4 and `T` is in

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think "power of 2" is what we want here, as that is how (generalized) quaternion groups are defined. For the function that does the exact same construction for "multiple of 4", there is dicyclic_group below. In particular, this distinction in the docstring matches to how is_quaternion_group works

Copy link
Collaborator

@thofma thofma Feb 27, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hm, I think that the consensus in #1630 is that both should be the same. Here are some reasons why I think this is the case:

  1. The definition with 2-powers is unnecessarily restrictive. There are plenty of references that construct quaternion groups of order 4n.
  2. There is now already code out there that is using the most general definition (that the order is any number $4n$), because this is what people were being told in Dicyclic groups #1630
  3. Since we don't do any quaternion_group(k) -> group of order 2^k or 2^(k+2) shenanigans, there is no reason to restrict to 2-powers.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah I decided to just treat both commands as exact equals, any artificial restriction benefits nobody but can cause frustration

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Note that e.g. Wikipedia allows "generalized quaternion groups" to have order $4k$.

@@ -827,6 +827,8 @@ as an instance of `T`,
where `n` is a power of 2 and `T` is in
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think "power of 2" is what we want here, as that is how (generalized) quaternion groups are defined. For the function that does the exact same construction for "multiple of 4", there is dicyclic_group below. In particular, this distinction in the docstring matches to how is_quaternion_group works

@fingolfin fingolfin added the enhancement New feature or request label Feb 27, 2025
@jamesnohilly jamesnohilly marked this pull request as ready for review March 3, 2025 17:29
@lgoettgens lgoettgens requested review from fingolfin and thofma March 12, 2025 12:56
Copy link

codecov bot commented Mar 12, 2025

Codecov Report

Attention: Patch coverage is 90.19608% with 5 lines in your changes missing coverage. Please review.

Project coverage is 84.38%. Comparing base (4bccbc1) to head (05ac3ec).

Files with missing lines Patch % Lines
src/Groups/group_constructors.jl 90.19% 5 Missing ⚠️
Additional details and impacted files
@@            Coverage Diff             @@
##           master    #4661      +/-   ##
==========================================
- Coverage   84.38%   84.38%   -0.01%     
==========================================
  Files         673      673              
  Lines       90284    90287       +3     
==========================================
+ Hits        76185    76186       +1     
- Misses      14099    14101       +2     
Files with missing lines Coverage Δ
src/Groups/group_constructors.jl 93.49% <90.19%> (+0.11%) ⬆️

... and 1 file with indirect coverage changes

🚀 New features to boost your workflow:
  • Test Analytics: Detect flaky tests, report on failures, and find test suite problems.

Copy link
Collaborator

@thofma thofma left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Very nice, thanks. But some group theory person should have a look too.


Return the dicyclic group of order `n`, as an instance of `T`,
where `n` is a multiple of 4 and `T` is in
{`PcGroup`, `SubPcGroup`, `PermGroup`,`FPGroup`, `SubFPGroup`}.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Adjust this to fully mirror the quaternion_group docstring (including a @ref link in the other direction).

... or... perhaps we should instead really merge the two, like GAP does: show a single docstring that starts with

"""
    dicyclic_group(::Type{T} = PcGroup, n::IntegerUnion)
    quaternion_group(::Type{T} = PcGroup, n::IntegerUnion)

...

Then attach that to, say, quaternion_group, and turn dicyclic_group into a "true" alias by doing const dicyclic_group = quaternion_group (I think it'll then get the same docstring). Merge is_quaternion_group and is_dicyclic_group` similarly.

In the examples use only one (quaternion).

Then instead of This is an alias of dicyclic_group. write something more like this:

!!! note
    For historical reasons and backwards compatibility, `dicyclic_group` is an alias
    of `quaternion_group`. The two functions are fully identical. We recommend always
    using `quaternion_group`.

and then similar for the is_* function.

Thoughts @ThomasBreuer @thofma ?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If it is just an alias, as a user I would not quite understand why you provide an alias that I am happy to find (because "dicyclic group" is the right name in my domain) and use, but then tell me not to use it.

But I don't have a strong opinion on this.

Copy link
Member

@ThomasBreuer ThomasBreuer Mar 21, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

My understanding of the discussion from issue #1630 is that it should be the other way round, that is, we recommend always using dicyclic_group.

And yes, I think it makes sense to have just one docstring for both names.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I am fine with not making a recommendation, my main concern is that I think it'd be nice to mention why we have two names.

true
```
"""
@gapattribute is_quaternion_group(G::GAPGroup) = GAP.Globals.IsQuaternionGroup(GapObj(G))::Bool
@gapattribute is_dicyclic_group(G::GAPGroup) =
GAP.Globals.IsQuaternionGroup(GapObj(G))::Bool
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I am getting confused.
GAP does the following:

gap> G:= DicyclicGroup(24);;
gap> IsQuaternionGroup(G);
false
gap> G:= DicyclicGroup(32);;
gap> IsQuaternionGroup(G);
true

That is, both QuaternionGroup and DicyclicGroup can be used to create the groups in question,
but IsQuaternionGroup regards only the dicyclic groups of 2-power order as (generalized) quaternion groups.
In fact, GAP seems to have no function that checks whether a group is dicyclic.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As far as I see, the GAP function DoComputeGeneralisedQuaternionGenerators (which is called by IsQuaternionGroup) would work as an is_dicyclic check if one just omits the check for 2-power order.

@ThomasBreuer
Copy link
Member

How shall we proceed?
Meanwhile gap-system/gap/pull/5966 got merged, do we want to wait until Oscar can use it, or do we want to add Oscar code that provides the same functionality?

@jamesnohilly
Copy link
Collaborator Author

I have now merged the docstrings of dicyclic_group and quaternion_group, with main preference on dicyclic_group as in #1630.

As for the is_{dicyclic,quaternion}_group issue, we could wait for a new GAP.jl release to use these new methods, however I am unsure. Adding this functionality to OSCAR and then eventually replacing is an option aswell.

Comment on lines +898 to +899
n = N//2
a = n//2
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This creates rationals but since the numbers are even we are allowed to divide, so we should be able to do this:

Suggested change
n = N//2
a = n//2
n = quo(N,2)
a = quo(n,2)

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think you mean div instead of quo

a = n//2

G1, _ = derived_subgroup(G)
!(is_cyclic(G1) && order(G1) == a) && return false
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This may be a matter of taste, so take with a grain of salt; but I find this easier to understand:

Suggested change
!(is_cyclic(G1) && order(G1) == a) && return false
(is_cyclic(G1) && order(G1) == a) || return false

I read this as "either G1 is cyclic and has order a, or else we return false

Feelfree to ignore this!

T = right_transversal(G, G1)
i = 1
while true
H = GAP.Globals.ClosureGroup(GapObj(G1), GapObj(T[i]))
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This line motivated me to write issue #4777 ;-)

Comment on lines +906 to +911
i = 1
while true
H = GAP.Globals.ClosureGroup(GapObj(G1), GapObj(T[i]))
Zn, _ = Oscar._as_subgroup(G1, H)
i = i + 1
if (i > 4) || (is_cyclic(Zn) && order(Zn) == n)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I guess we could do away with i and just do this? But it would also make the code differ more from the GAP version, so perhaps not worth it anyway

Suggested change
i = 1
while true
H = GAP.Globals.ClosureGroup(GapObj(G1), GapObj(T[i]))
Zn, _ = Oscar._as_subgroup(G1, H)
i = i + 1
if (i > 4) || (is_cyclic(Zn) && order(Zn) == n)
@assert length(T) == 4
for x in T
H = GAP.Globals.ClosureGroup(GapObj(G1), GapObj(x))
Zn, _ = Oscar._as_subgroup(G1, H)
if is_cyclic(Zn) && order(Zn) == n

Comment on lines +834 to +835
of `dicyclic_group`. The two functions are fully identical. We recommend always
using `dicyclic_group`.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
of `dicyclic_group`. The two functions are fully identical. We recommend always
using `dicyclic_group`.
of `dicyclic_group`. The two functions are fully identical.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement New feature or request topic: groups
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Dicyclic groups
5 participants