-
Notifications
You must be signed in to change notification settings - Fork 162
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 SemidirectDecompositions and enhance/change StructureDescription #763
Add SemidirectDecompositions and enhance/change StructureDescription #763
Conversation
62534c1
to
8377194
Compare
Of course, |
For comparison:
Further, old
The smallest example of the behaviour change due to using
For this group without #563 the new method errors out with "No method found".
|
8377194
to
32e5719
Compare
Further, the new method significantly boosts the groups
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
All in all, this looks like a very desirable change. I have only minor nitpicks, so basically I think this could be merged right away.
However, if you were willing to rebase this on current master, then codecov would get a chance to run on this, and we could see if the tests exercise all cases of your code.
Thank you very much!
"C2 x (C3 : C4)", "(C6 x C2) : C2", "C12 x C2", "C3 x D8", "C3 x Q8", | ||
"S4", "C2 x A4", "C2 x C2 x S3", "C6 x C2 x C2" ], [ "C25", "C5 x C5" ], | ||
"C2 x (C3 : C4)", "C3 : D8", "C12 x C2", "C3 x D8", "C3 x Q8", "S4", | ||
"C2 x A4", "C2 x C2 x S3", "C6 x C2 x C2" ], [ "C25", "C5 x C5" ], | ||
[ "D26", "C26" ], | ||
[ "C27", "C9 x C3", "(C3 x C3) : C3", "C9 : C3", "C3 x C3 x C3" ], | ||
[ "C7 : C4", "C28", "D28", "C14 x C2" ], [ "C29" ], |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Reading this diff is really nasty... which is not your fault, mind you! I think we should reformat NAMES_OF_SMALL_GROUPS so that there is exactly one string on each line.
But if I do that now, it would break this PR, so I am not going to do that on master for now. (If you feel bored and generous, feel free to do it as part of your PR, but then it only makes sense if you do it "twice": once in a commit before your change, were just the existing names are reformatted, and then of course after your changes, so that one can see the actual diffs in the relevant commit. Doing that of course is some work, so I am not asking you to perform that, nor am I going to hold your PR hostage over this! It's merely an idea)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I do not mind doing this, but I am not entirely sure what to do. I am not even sure how to rebase.
Is the rebase simply to go into my SemidirectDecompositions
branch, and add the command
git rebase gap-system/master
(after of course fetching gap-system/master)?
And if this is done, then how should I add an extra commit where I reformat NAMES_OF_SMALL_GROUPS? Should I somehow go to an earlier commit with some git command, do the change, and commit?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ok, I did
git rebase gap-system/master
, then
git status
gave me this:
On branch SemidirectDecompositions
Your branch and 'origin/SemidirectDecompositions' have diverged,
and have 303 and 4 different commits each, respectively.
(use "git pull" to merge the remote branch into yours)
nothing to commit, working directory clean
Then I did
git checkout becaeff915dbaef
, and now I am stuck with the message
Note: checking out 'becaeff915dbaef'.
You are in 'detached HEAD' state. You can look around, make experimental
changes and commit them, and you can discard any commits you make in this
state without impacting any branches by performing another checkout.
If you want to create a new branch to retain commits you create, you may
do so (now or later) by using -b with the checkout command again. Example:
git checkout -b new_branch_name
HEAD is now at becaeff... Merge pull request #936 from markuspf/globalstate-fixes
I do not dare to do anything at this point until someone wiser suggests how to move forward instead of doing something irreversible....
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why did you do that checkout? Anyway, it makes little sense to me. What you want instead is to again checkout your local branch SemidirectDecompositions
. That's the result of your rebase. It now diverse from origin/SemidirectDecompositions
, i.e. what is on the server - that's normal and to be expected, as you performed a rebase. The larger number 303 of additional commits in the local branch SemidirectDecompositions
stems from the fact that it now also includes all the work that happened on the master branch since april...
So, this work should now be pushed to your github clone of gap, which will then be reflected in this PR.
This should do it:
git checkout SemidirectDecompositions
git push --force
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
One nice way to "insert" commits (or also edit, modify ec.) on a branch which is still work-in-progress (i.e. has not yet been merged into, say, master) is to use "interactive rebase". Here are some hints on that: https://robots.thoughtbot.com/git-interactive-rebase-squash-amend-rewriting-history
But don't worry too much about this detail, it's in the category of "would be kinda nice, but in the end is unimportant".
Though I still recommend learning about interactive rebase, it's one of my favorite git features, I use it constantly for my branches
"C22 x C2 x C2" ], [ "C89" ], | ||
[ "C5 x D18", "C9 x D10", "D90", "C90", "C3 x C3 x D10", "C15 x S3", | ||
"C3 x D30", "C5 x ((C3 x C3) : C2)", "(C15 x C3) : C2", "C30 x C3" ], | ||
"C3 x D30", "C5 x ((C3 x C3) : C2)", "(C3 x C3) : D10", "C30 x C3" ], | ||
[ "C91" ], [ "C23 : C4", "C92", "D92", "C46 x C2" ], [ "C31 : C3", "C93" ], | ||
[ "D94", "C94" ], [ "C95" ],, [ "C97" ], | ||
[ "D98", "C98", "C7 x D14", "(C7 x C7) : C2", "C14 x C7" ], |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Many of these changes look like improvements to me; some perhaps are slightly worse -- but I think this is mostly subjective. In any case, I didn't notice anything that struck me as "bad", so I am fine with that part of the change.
@@ -267,17 +268,56 @@ DeclareGlobalFunction( "DirectFactorsOfGroupKN", IsGroup ); | |||
|
|||
############################################################################# | |||
## | |||
#A SemidirectFactorsOfGroup( <G> ) . decomposition into a semidirect product |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I was very mildly concerned about the removal (resp. renaming of SemidirectFactorsOfGroup
). However, it was undocumented, and no package I am aware of uses it. So, in the end I have no objections to that.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I did a grep on the gap library for this undocumented function (no hits), and also asked @Stefan-Kohl. He advised to simply remove the old code.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We are in full agreement then 😀 (note the past tense in my comment).
@@ -646,13 +686,14 @@ DeclareGlobalFunction( "LinearGroupParameters" ); | |||
## <Mark>1.</Mark> | |||
## <Item> | |||
## Lookup in a precomputed list, if the order of <A>G</A> is not | |||
## larger than 100 and not equal to 64. | |||
## larger than 100 and not equal to 64 or 96. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ahh, good catch :)
fi; | ||
return NHs; | ||
fi; | ||
end); | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I have not reviwed this code in detail, but on a cursory glance it looks sane, and I trust that you know what you are doing here... :-)
@@ -1557,6 +1635,8 @@ InstallMethod( StructureDescription, | |||
" x ","")); | |||
fi; | |||
|
|||
if not IsFinite(G) then TryNextMethod(); fi; | |||
|
|||
# special case alternating group |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
OK
#[ "6", "S3", "D12", "A4", "3xS3", "2xA4", "S4", "S4", "S3xS3", "(3^2):4", | ||
# "2xS4", "A5", "(S3xS3):2", "S5", "A6", "S6" ] | ||
#gap> StructureDescription(PSL(4,2)); | ||
#"A8" | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
OK, so just remove that chunk (instead of commenting it out)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ok, just removed in a new commit.
## gap> G := Group([ (6,7,8,9,10), (8,9,10), (1,2)(6,7), (1,2,3,4,5)(6,7,8,9,10) ]);; | ||
## gap> StructureDescription(G); | ||
## "A5 : S5" | ||
## |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is there a point to this commented out example? Is it too slow or what? Or why else is it commented out? I'd say, either remove this; or uncomment it; or else add a brief blurb that explains why it is commented out.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, there is. Currently this example works. After merging this PR, this example will not work. The reason is that the new SemidirectDecompositions
code uses ComplementClassesRepresentatives
which does not work if both N and G/N is non-solvable. A proposed solution and a discussion about this arose in my PR #563. So this is a reminder to all of us that this example breaks after the PR is merged.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Added an explanation into the file.
|
||
## | ||
##gap> StructureDescription(AbelianGroup([0,2,3,4,5,6,7,8,9,10]):short); | ||
##"0x2520x60x6x2x2" |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hmm... see above
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah, yes, this example would fail, because the value short
does not currently work for infinite Abelian groups. For that to work, some essential change in the coding should be done, which I did not care to do at the time for mostly two reasons: this feature was not supported previously, anyway, and it would have taken me a longer time. Anyway, so this serves me as a reminder to do this at a later point. If it bothers you, I can remove it.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Added an explanation into the file.
gap> StructureDescription(testG(8,3)); | ||
"C3 x QD16" | ||
gap> StructureDescription(testG(8,4)); | ||
"(C16 x C4) : C2" | ||
"((C16 x C2) : C2) : C2" |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ahh, too bad, the old ones looked "nicer" to me... but ignore me, it is probably impossible to making a "perfect" choice here (even if one assumed that all mathematicians had the same "taste", which I am sure we don't :-). Short of hard coding lots of cases, that is.
Current coverage is 49.44% (diff: 100%)@@ master #763 diff @@
==========================================
Files 424 424
Lines 222845 222948 +103
Methods 3430 3430
Messages 0 0
Branches 0 0
==========================================
+ Hits 110120 110230 +110
+ Misses 112725 112718 -7
Partials 0 0
|
90d64ab
to
db5814b
Compare
Thanks for rebasing! The code coverage report suggests that all your new code (with the exception of one call to |
db5814b
to
e4e5cb5
Compare
@fingolfin I think I have managed to reformat |
Hm, Anybody has any suggestions? |
@hungaborhorvath : I strongly suspect it is not your fault -- this test occasionally fails for unpredicatable reasons. I've told travis to try re-running the test, to see if it fixes it. |
@ChrisJefferson You were right, apparently now travis succeeded. |
Weird: When I run your new tests manually, I get a test failure:
|
@fingolfin After some playing around I figured out the reason: the elements of the list |
Well, I don't mind the diferent descriptions much. But this is an instability in the tests which can cause us some headaches in the future. It seems the test results differ whether packages are loaded or not, which is an issue, as @alex-konovalov will likely confirm ;-). So, something should be done after all. As to whether (and what) can be done... One easy (but not very nice) option would to be modify the tests to skip these particular groups. Another might be to sort the normal subgroups. Now, a full sort is potentially slow, but it might suffice to sort them by size. |
return 0; | ||
end; | ||
if not IsBound(Ns) then | ||
Ns := ShallowCopy(NormalSubgroups(G)); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Regarding the tests differeing on which packages are present: Perhaps it would suffice to insert this here (have not yet had time to try it):
SortBy(Ns, Size);
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@fingolfin @alex-konovalov I have tried it with -r -A -x 80 -m 100m -o 1g. It is not enough. For one, SmallGroup(16, 13)
(and a lot of other groups of size <100) have different StructureDescription
with and without packages. I could try to sort the NormalSubgroups
completely according to some general way, but that would defeat efficiency. And I am not entirely sure how to do it, either.
Second, in StructureDescription.tst:25 the group will have A5 x A5
both with and without packages, but if no packages are present then maxsub.gi:545 will be invoked, and an info message "Alternating recognition needed" will be printed along with the output. Not sure if that can be nicely reconciled.
Finally, without packages the test runs in 8 seconds on my laptop, with packages it runs for 15 seconds. I am wondering if this is really how it should be....
I am not sure how to proceed at this point.
PS. Now that #563 is merged, I think I should rebase according to new master and uncomment the smallest example A5 : S5
in the tests. Would that be ok, or should I wait until the current problems are sorted?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@fingolfin @alex-konovalov Ok, rebased according to newest master and moved the slower tests to tst/teststandard/opers
I also added a slow (about 1 min) test checking if grpnames.g
contains the correct StructureDescription
for the groups of size at most 100 (excluding 64 and 96).
No idea what to do with the discrepancies without packages, including the info message "Alternating recognition needed" and the differently ordered NormalSubgroups
. Just ordering normal subgroups by size is not enough, a complete ordering defeats efficiency, removing the tests in question defeats the testing purpose.
Finally the following example (A5:S5) takes half time without packages (6 seconds without packages, 13 seconds with packages):
G := Group([ (6,7,8,9,10), (8,9,10), (1,2)(6,7), (1,2,3,4,5)(6,7,8,9,10) ]);;
ConjugacyClassesSubgroups(G);;
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@hungaborhorvath Thanks for the rebase.
As long as packages add new methods for things like NormalSubgroups
, we will never be able to guarantee that these descriptions remain stable. And of course you are right, fully ordering the normal subgroups would be foolish in general.
So, perhaps we should simply stop any pretense that there is stability, and thus remove the relevant tests. Or only run them when the right packages are (not) loaded.
Another option would be to add another option, say stable
or fullyinvariant
or so to StructureDescription
which instructs it to sort NormalSubgroups
(assuming that is even enough). Then, compute the precomputed descriptions using that option, and use that option for the tests that verify the precomputed values. And indeed, do not run this test as part of the standard test suite. (I am not even sure it should be a regular test... Perhaps it would be better to have a function which (re)generates the file containing the precomputed names. Then, one can run that manually from time to time, if one is concerned about behavior changes).
Tests which fail at the slightest change of things are useless for us; esp. if mathemtically valid improvements cause these failures. Also, we never guarantee that structure descriptions are unique or never change, do we?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@fingolfin Hm, yes, maybe I should add an extra option stable
or nicest
or something. Actually, that way I could bring the old behaviour back for whoever still wants GAP to go through all possible descriptions and choose one by some rules (probably by the old rules just to keep consistent with old behaviour).
Another possibility that came to my mind is to have an option, say full
, describing all possible descriptions (provided by the current code) into a set, and that would make tests completely stable. Using this we could rewrite our current tests such that if the result is an element of the full set or not. These probably would be long tests, maybe they should go into teststandard rather than testinstall?
For these changes I will need some time, though.
291e923
to
561d109
Compare
@fingolfin I have stabilized the tests by adding back the old functionality under the option "nice". I have updated the documentation accordingly. The file Further, the "short" option is enhanced, it is faster than it was before, recognizes nonprimepower duplicates in the cyclic decomposition (C6 x C6 is now shortly 6^2) and also works for infinite abelian groups. I believe the tests in At some later date I might write an option to determine all sensible descriptions, but not in the foreseeable future. Therefore I would rather have this merged if everybody is happy with it, and later I will do another PR for the new option. |
Ok, so now appveyor failed, and I have no idea what that is.... Can someone help me what I messed up here? Thank you. |
@hungaborhorvath : Just ignore that failure, it's a random computer hiccup. |
@hungaborhorvath I've restarted the AppVeyor build, but from the output if was obvious that the cygwin64 build failed even before starting to compile GAP, so that was obviously not your fault. For OS-dependent PR it's better not to ignore failed AppVeyor builds though. |
@alex-konovalov This could be a short description into the 4.9.0 changelog:
|
A side note: the undocumented new attribute and some new undocumented functions are fully documented in |
@hungaborhorvath: My suggestion is to merge #965 first (I am testing it locally at the moment, so hopefully today) to ensure that tests are runnable, and then merge this PR of yours before it will require another rebase. Thanks for patience and polishing all tests! |
@alex-konovalov Thank you. After #965 is merged I believe I will need to slightly alter the test files because of the InfoWarning <-> InfoPerformance change. |
!!! WARNING !!! This changes the behaviour of StructureDescription compared to the previous state, which was inconsistent with the manual. SemidirectDecompositionsOfFiniteGroup is added. This computes all or some semidirect decompositions of a group. An optional second argument can be given as the normal subgroups for which complements should be sought. Complements for normal subgroups are found by ComplementClassesRepresentatives. Unfortunately, this does not work if both N and G/N are nonsolvable. Further, an optional argument "any" is recognized, when it should return only one nontrivial semidirect decomposition if exists and fail otherwise. This is mostly implemented by applying Schur-Zassenhaus by checking if any nontrivial normal Hall subgroups exist. Another optional argument "str" is recognized, when it does not necessarily compute the complement to the normal subgroup if Schur-Zassenhaus applies. The reason is that computing the complement might be expensive and for StructureDescription only the isomorphism type of the complement is interesting. SemidirectDecompositions is a new attribute, computing all semidirect decompositions. SemidirectFactorsOfGroup is removed completely. It computed all conjugacy classes of all subgroups and therefore was rather inefficient. Further, it did not compute _all_ semidirect decompositions, but only the ones with subgroups having the same size as the first subgroup having a normal complement. This yielded inconsistent behaviour with the manual, the smallest example for such a group was SmallGroup(504,7). StructureDescription now works for infinite abelian groups, as well. Further, for semidirect decompositions it calls SemidirectDecompositionsOfFiniteGroup with "str" argument. NOTE that if the group G has a nonsolvable normal subgroup N such that G/N is nonsolvable, as well, then ComplementClassesRepresentatives errors with "No method found", and thus so do SemidirectDecompositions and StructureDescription. Previously computations for such groups did not error with "No method found", but probably took a long time. gap-system#563 tries to remedy this using the old and inefficient method, however whether it should be merged or not is currently under debate. In the meantime, for such groups we throw an error. StructureDescription of groups having size at most 100 are recomputed, and the manual of StructureDescription is rewritten to match current behaviour. Tests are added. All lines of SemidirectDecompositions are run. Not all lines of StructureDescription are run, some examples for the missing lines should be found later. Tests are aligned with the new behaviour, and not the old one.
This commit adds the old documented behaviour of StructureDescription. If option nice is set, then all SemidirectDecompositions are computed and one [N,H] will be chosen by the old preferences, namely abelian H, abelian N with the most abelian invariants and the most direct factors, and with H acting faithfully on N.
Further, grpnames.g contains the "nice" outputs of StructureDescription.
Now the option short works for infinite abelian groups. Further, it now recognizes nonprimepower duplicates, as well. E.g. the short StructureDescription for the group C6 x C6 is now 6^2.
ce20d70
to
0ee8cf5
Compare
@alex-konovalov I rebased my PR to new master, otherwise tests would not have run because of the I manually tested the relevant files with and without packages, and squashed my change into one of my earlier commits, just to keep the history (sort of) clean. |
@hungaborhorvath thanks. I am not sure that remembering the InfoLevel in the test is the right solution - this message occurs in many tests, and doing the same trick in each test seem not to be the right thing. Maybe FYI, there are only 7 messages in |
@hungaborhorvath FYI,
|
@alex-konovalov Shoot... In the end I have forgotten to properly update the examples in the manual (BTW, all examples should be in the testinstall/opers/StructureDesctiption file). In fact, I also wanted to add an example about how to use the "nice" option. Further, I understand that now starting tests change InfoPeformance to 0, so I probably should remove those lines from my tests, as well. I will put a new PR with these changes soon. |
Further, correct the old examples from gap-system#763. Add some extra tests. Remove InfoPerformance from tst files as per 392ad82.
!!! WARNING !!! This changes the behaviour of StructureDescription
compared to the previous state, which was inconsistent with the manual.
SemidirectDecompositionsOfFiniteGroup is added. This computes all or some
semidirect decompositions of a group. An optional second argument can be
given as the normal subgroups for which complements should be sought.
Complements for normal subgroups are found by
ComplementClassesRepresentatives. Unfortunately, this does not work if
both N and G/N are nonsolvable.
Further, an optional argument "any" is recognized, when it should return
only one nontrivial semidirect decomposition if exists and fail otherwise.
This is mostly implemented by applying Schur-Zassenhaus by checking if
any nontrivial normal Hall subgroups exist. Another optional argument
"str" is recognized, when it does not necessarily compute the complement
to the normal subgroup if Schur-Zassenhaus applies. The reason is that
computing the complement might be expensive and for StructureDescription
only the isomorphism type of the complement is interesting.
SemidirectDecompositions is a new attribute, computing all semidirect
decompositions.
SemidirectFactorsOfGroup is removed completely. It computed all conjugacy
classes of all subgroups and therefore was rather inefficient. Further, it
did not compute all semidirect decompositions, but only the ones with
subgroups having the same size as the first subgroup having a normal
complement. This yielded inconsistent behaviour with the manual, the
smallest example for such a group was SmallGroup(504,7).
StructureDescription now works for infinite abelian groups, as well.
Further, for semidirect decompositions it calls
SemidirectDecompositionsOfFiniteGroup with "str" argument.
NOTE that if the group G has a nonsolvable normal subgroup N such that
G/N is nonsolvable, as well, then ComplementClassesRepresentatives errors
with "No method found", and thus so do SemidirectDecompositions and
StructureDescription. Previously computations for such groups did not error
with "No method found", but probably took a long time. #563 tries to
remedy this using the old and inefficient method, however whether it should
be merged or not is currently under debate. In the meantime, for such
groups we throw an error.
StructureDescription of groups having size at most 100 are recomputed, and the manual of StructureDescription is rewritten to match current behaviour.
Tests are added. All lines of SemidirectDecompositions are run. Not all
lines of StructureDescription are run, some examples for the missing lines
should be found later.
Tests are aligned with the new behaviour and not the old one.
Edit: The old documented behaviour of StructureDescription is still available.
If option nice is set, then all SemidirectDecompositions are computed
and one [N,H] will be chosen by the old preferences, namely abelian H,
abelian N with the most abelian invariants and the most direct factors,
and with H acting faithfully on N.