forked from libsemigroups/libsemigroups_pybind11
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathtodd-coxeter.cpp
More file actions
627 lines (512 loc) · 22.9 KB
/
todd-coxeter.cpp
File metadata and controls
627 lines (512 loc) · 22.9 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
//
// libsemigroups_pybind11
// Copyright (C) 2025 James Mitchell
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
//
// libsemigroups headers
#include <libsemigroups/todd-coxeter.hpp>
// pybind11....
#include <pybind11/chrono.h>
#include <pybind11/pybind11.h>
#include <pybind11/stl.h>
// libsemigroups_pybind11....
#include "cong-common.hpp" // for def_construct_default
#include "main.hpp" // for init_todd_coxeter
namespace libsemigroups {
namespace py = pybind11;
using std::literals::operator""sv;
namespace {
template <typename Word>
void bind_todd_coxeter(py::module& m, std::string const& name) {
using ToddCoxeter_ = ToddCoxeter<Word>;
py::class_<ToddCoxeter_, detail::ToddCoxeterImpl> thing(m,
name.c_str(),
R"pbdoc(
Class containing an implementation of the Todd-Coxeter Algorithm.
This class contains an implementation of the Todd-Coxeter algorithm for
computing 1-sided (right), and 2-sided congruences on a semigroup or monoid.
In this documentation we use the term "congruence enumeration" to mean the
execution of (any version of) the Todd-Coxeter algorithm.
.. seealso :any:`congruence_kind`, :any:`tril`, and :any:`Runner`.
.. |name| replace:: :any:`ToddCoxeter`
.. include:: ../../../_static/runner_non_inherit.rst
.. doctest::
>>> from libsemigroups_pybind11 import (presentation, Presentation, ToddCoxeter,
... congruence_kind, word_graph, Order, todd_coxeter)
>>> tc = ToddCoxeter(word=str)
>>> options = tc.options
>>> p = Presentation("ab")
>>> p.contains_empty_word(True)
<monoid presentation with 2 letters, 0 rules, and length 0>
>>> presentation.add_rule(p, "aa", "")
>>> presentation.add_rule(p, "a", "b")
>>> tc.init(congruence_kind.onesided, p).strategy(options.strategy.felsch)
<1-sided ToddCoxeter over <monoid presentation with 2 letters, 2 rules, and length 4> with 0 gen. pairs + 1 node>
>>> tc.number_of_classes()
2
>>> tc.contains("aaaa", "aa")
True
>>> tc.index_of("aaaa")
0
>>> p = Presentation("abcd")
>>> presentation.add_rule(p, "aa", "a");
>>> presentation.add_rule(p, "ba", "b");
>>> presentation.add_rule(p, "ab", "b");
>>> presentation.add_rule(p, "ca", "c");
>>> presentation.add_rule(p, "ac", "c");
>>> presentation.add_rule(p, "da", "d");
>>> presentation.add_rule(p, "ad", "d");
>>> presentation.add_rule(p, "bb", "a");
>>> presentation.add_rule(p, "cd", "a");
>>> presentation.add_rule(p, "ccc", "a");
>>> presentation.add_rule(p, "bcbcbcbcbcbcbc", "a");
>>> presentation.add_rule(p, "bcbdbcbdbcbdbcbdbcbdbcbdbcbdbcbd", "a");
>>> tc = ToddCoxeter(congruence_kind.twosided, p)
>>> tc.strategy(options.strategy.hlt).lookahead_extent(options.lookahead_extent.partial).save(False)
<2-sided ToddCoxeter over <semigroup presentation with 4 letters, 12 rules, and length 79> with 0 gen. pairs + 1 node>
>>> tc.number_of_classes()
10752
>>> tc
<2-sided ToddCoxeter over <semigroup presentation with 4 letters, 12 rules, and length 79> with 0 gen. pairs + 10,753 nodes>
>>> tc.word_graph()
<WordGraph with 10,753 nodes, 43,012 edges, & out-degree 4>
>>> it = todd_coxeter.normal_forms(tc)
>>> [next(it) for _ in range(10)]
['a', 'b', 'c', 'd', 'bc', 'bd', 'cb', 'db', 'bcb', 'bdb']
>>> tc.standardize(Order.lex)
True
>>> it = todd_coxeter.normal_forms(tc)
>>> [next(it) for _ in range(10)]
['a', 'ab', 'abc', 'abcb', 'abcbc', 'abcbcb', 'abcbcbc', 'abcbcbcb', 'abcbcbcbc', 'abcbcbcbcb']
)pbdoc");
thing.def("__repr__", [](ToddCoxeter_ const& self) {
return to_human_readable_repr(self);
});
////////////////////////////////////////////////////////////////////////
def_construct_default(thing, "ToddCoxeter");
def_init_default(thing, "ToddCoxeter");
def_construct_kind_presentation(thing, "ToddCoxeter");
def_init_kind_presentation(thing, "ToddCoxeter");
def_copy(thing, "ToddCoxeter");
// number_of_classes is implemented in ToddCoxeterImpl
def_generating_pairs(thing, "ToddCoxeter");
def_presentation(thing,
"ToddCoxeter",
doc{.detail =
R"pbdoc(If the :any:`ToddCoxeter` instance was
constructed or initialised from a :any:`WordGraph`, then this presentation
will be empty.)pbdoc"sv});
def_add_generating_pair(thing, "ToddCoxeter");
def_currently_contains(thing, "ToddCoxeter");
def_contains(thing, "ToddCoxeter");
def_reduce_no_run(thing, "ToddCoxeter", doc{.detail = R"pbdoc(
If the :any:`ToddCoxeter` instance is not :any:`Runner.finished`,
then it might be that equivalent input words produce different output
words. This function triggers no congruence enumeration.)pbdoc"sv});
def_reduce(thing, "ToddCoxeter");
////////////////////////////////////////////////////////////////////////
thing.def(py::init<congruence_kind, ToddCoxeter_ const&>(),
py::arg("knd"),
py::arg("tc"),
R"pbdoc(
:sig=(self: ToddCoxeter, knd: congruence_kind, tc: ToddCoxeter) -> None:
Construct from :any:`congruence_kind` and :any:`ToddCoxeter`.
This function constructs a :any:`ToddCoxeter` instance representing a
congruence of kind *knd* over the :any:`ToddCoxeter` instance *tc*. The
:any:`ToddCoxeter` instance constructed in this way represents a quotient of
the word graph represented by *tc*.
:param knd: the kind (onesided, or twosided) of the congruence.
:type knd: congruence_kind
:param tc: the :any:`ToddCoxeter` instance.
:type tc: ToddCoxeter
:raises LibsemigroupsError:
if the arguments *knd* and *tc* are not compatible. If the first item is
``tc.kind()`` and the second is the parameter *knd*, then compatible
arguments are (one-sided, one-sided), (two-sided, one-sided), and (two-sided,
two-sided).)pbdoc");
thing.def(
"init",
[](ToddCoxeter_& self, congruence_kind knd, ToddCoxeter_ const& tc)
-> ToddCoxeter_& { return self.init(knd, tc); },
py::arg("knd"),
py::arg("tc"),
R"pbdoc(
:sig=(self: ToddCoxeter, knd: congruence_kind, tc: ToddCoxeter) -> ToddCoxeter:
Re-initialize a ToddCoxeter instance.
This function puts a :any:`ToddCoxeter` instance back into the state
that it would have been in if it had just been newly constructed from
*knd* and *tc*.
:param knd: the kind (onesided, or twosided) of the congruence.
:type knd: congruence_kind
:param tc: the :any:`ToddCoxeter` instance.
:type tc: ToddCoxeter
:returns: *self*.
:rtype: ToddCoxeter
:raises LibsemigroupsError:
if the arguments *knd* and *tc* are not compatible. If the first item is
``tc.kind()`` and the second is the parameter *knd*, then compatible
arguments are (one-sided, one-sided), (two-sided, one-sided), and (two-sided,
two-sided).
)pbdoc");
thing.def(py::init<congruence_kind, WordGraph<uint32_t> const&>(),
py::arg("knd"),
py::arg("wg"),
R"pbdoc(
:sig=(self: ToddCoxeter, knd: congruence_kind, wg: WordGraph) -> None:
Construct from :any:`congruence_kind` and :any:`WordGraph`.
This function constructs a :any:`ToddCoxeter` instance representing a
congruence of kind *knd* over the :any:`WordGraph` *wg*. The
:any:`ToddCoxeter` instance constructed in this way represents a
quotient of the word graph *wg*. If *wg* happens to be the left
or right Cayley graph of a semigroup or monoid, then the
:any:`ToddCoxeter` instance will represent a quotient of that
semigroup.
:param knd: the kind (onesided or twosided) of the congruence.
:type knd: congruence_kind
:param wg: the word graph.
:type wg: WordGraph
)pbdoc");
thing.def(
"init",
[](ToddCoxeter_& self,
congruence_kind knd,
WordGraph<uint32_t> const& wg) -> ToddCoxeter_& {
return self.init(knd, wg);
},
py::arg("knd"),
py::arg("wg"),
R"pbdoc(
:sig=(self: ToddCoxeter, knd: congruence_kind, wg: WordGraph) -> ToddCoxeter:
Construct from :any:`congruence_kind` and :any:`WordGraph`.
This function constructs a :any:`ToddCoxeter` instance representing a
congruence of kind *knd* over the :any:`WordGraph` *wg*. The
:any:`ToddCoxeter` instance constructed in this way represents a
quotient of the word graph *wg*. If *wg* happens to be the left
or right Cayley graph of a semigroup or monoid, then the
:any:`ToddCoxeter` instance will represent a quotient of that
semigroup.
:param knd: the kind (onesided or twosided) of the congruence.
:type knd: congruence_kind
:param wg: the word graph.
:type wg: WordGraph
)pbdoc");
thing.def(
"current_index_of",
[](ToddCoxeter_ const& self, Word const& w) {
return from_int(todd_coxeter::current_index_of(self, w));
},
py::arg("w"),
R"pbdoc(
:sig=(self: ToddCoxeter, w: list[int] | str) -> int | Undefined:
Returns the current index of the class containing a word.
This function returns the current index of the class containing the word
*w* No enumeration is triggered by calls to this function. Unless
:any:`Runner.finished` returns ``True``, the index returned by this
function is essentially arbitrary, and can only really be used to check
whether or not two words are currently known to belong to the congruence.
The returned index is obtained by following the path in
:any:`current_word_graph` from node ``0`` labelled by the word *w*. If
there is no such path, then :any:`UNDEFINED` is returned.
:param w: the word.
:type w: list[int] | str
:returns: The current index of the class containing the word.
:rtype: int | Undefined
:raises LibsemigroupsError:
if any of the values in *w* is out of range, i.e. they do not belong to
``presentation().alphabet()`` and ``Presentation.throw_if_letter_not_in_alphabet``
raises.
)pbdoc");
thing.def(
"index_of",
[](ToddCoxeter_& self, Word const& w) {
return todd_coxeter::index_of(self, w);
},
py::arg("w"),
R"pbdoc(
:sig=(self: ToddCoxeter, w: list[int] | str) -> int:
Returns the index of the class containing a word.
This function returns the index of the class containing the word *w* A
full enumeration is triggered by calls to this function. If the
:any:`current_word_graph` has not already been standardized, then this
function first standardizes it with respect to :any:`Order.shortlex`;
otherwise the existing standardization order is used. The returned index
is obtained by following the path in :any:`current_word_graph` from node
``0`` labelled by the word *w* Since a full enumeration is triggered by
calls to this function, the word graph is complete, and so the return
value is never :any:`UNDEFINED`.
:param w: the word.
:type w: list[int] | str
:returns: The index of the class containing the word.
:rtype: int
:raises LibsemigroupsError:
if any of the values in *w*, i.e. they do not belong to
``presentation().alphabet()`` and
``Presentation.throw_if_letter_not_in_alphabet`` raises.
)pbdoc");
thing.def(
"current_word_of",
[](ToddCoxeter_& self, size_t i) {
return todd_coxeter::current_word_of(self, i);
},
R"pbdoc(
:sig=(self: ToddCoxeter, i: int) -> list[int] | str:
Returns a current word representing a class with given index.
This function returns the current word representing the class with index *i*.
No enumeration is triggered by calls to this function, but
:any:`current_word_graph` is standardized (using :any:`Order.shortlex`) if it
is not already standardized. The output word is obtained by following a path in
:any:`current_spanning_tree` from the node corresponding to index *i* back to
the root of that tree.
:param i: the index of the class.
:type i: int
:returns: The word representing the *i*-th class.
:rtype: list[int] | str
:raises LibsemigroupsError: if *i* is out of bounds.
)pbdoc");
thing.def(
"word_of",
[](ToddCoxeter_& self, size_t i) {
return todd_coxeter::word_of(self, i);
},
py::arg("i"),
R"pbdoc(
:sig=(self: ToddCoxeter, i: int) -> list[int] | str:
Returns a word representing a class with given index.
This function returns the word representing the class with index *i*. A
full enumeration is triggered by calls to this function. The output word
is obtained by following a path in :any:`current_spanning_tree` from the
node corresponding to index *i* back to the root of that tree.
:returns: The word representing the *i*-th class.
:rtype: list[int]
:param i: the index of the class.
:type i: int
:raises LibsemigroupsError: if *i* is out of bounds.
)pbdoc");
////////////////////////////////////////////////////////////////////////
// Helpers from cong-common.hpp . . .
////////////////////////////////////////////////////////////////////////
auto raises = R"pbdoc(
:raises LibsemigroupsError:
if the number of classes in *tc* is infinite. In this case, the
enumeration of *tc* will not terminate successfully.)pbdoc"sv;
def_partition<ToddCoxeter<Word>>(
m,
"ToddCoxeter",
"todd_coxeter",
doc{.only_document_once = true, .raises = raises, .var = "tc"});
def_non_trivial_classes<ToddCoxeter<Word>>(
m,
"ToddCoxeter",
"todd_coxeter",
doc{.only_document_once = true, .raises = raises, .var = "tc"});
def_normal_forms<ToddCoxeter<Word>>(m,
"ToddCoxeter",
"todd_coxeter",
doc{.detail = R"pbdoc(
The order of the classes, and the normal forms, that are returned are
controlled by :any:`ToddCoxeter.standardize`. This function triggers a full
enumeration of ``tc``.)pbdoc",
.only_document_once = true,
.raises = raises,
.var = "tc"});
////////////////////////////////////////////////////////////////////////
// Helper functions - specific to ToddCoxeter
////////////////////////////////////////////////////////////////////////
m.def(
"todd_coxeter_redundant_rule",
[](Presentation<word_type> const& p, std::chrono::milliseconds t)
-> std::optional<std::pair<word_type, word_type>> {
auto it = todd_coxeter::redundant_rule(p, t);
if (it != p.rules.cend()) {
return std::make_pair(*it, *(it + 1));
}
return {};
},
py::arg("p"),
py::arg("t"),
R"pbdoc(
:sig=(p: Presentation, t: datetime.timedelta) -> tuple[list[int], list[int]] | tuple[str, str] | None:
:only-document-once:
Return a redundant rule or ``None``.
Starting with the last rule in the presentation, this function attempts to
run the Todd-Coxeter algorithm on the rules of the presentation except for a
given omitted rule. For every such omitted rule, Todd-Coxeter is run for the
length of time indicated by the second parameter *t*, and then it is checked
if the omitted rule can be shown to be redundant. If the omitted rule can be
shown to be redundant in this way, then this rule is returned. If no rule can
be shown to be redundant in this way, then ``None`` is returned.
:param p: the presentation.
:type p: Presentation
:param t: time to run Todd-Coxeter for every omitted rule.
:type t: datetime.timedelta
:returns: A redundant rule or ``None``.
:rtype: tuple[list[int], list[int]] | tuple[str, str] | None
)pbdoc");
m.def(
"todd_coxeter_redundant_rule",
[](Presentation<std::string> const& p, std::chrono::milliseconds t)
-> std::optional<std::pair<std::string, std::string>> {
auto it = todd_coxeter::redundant_rule(p, t);
if (it != p.rules.cend()) {
return std::make_pair(*it, *(it + 1));
}
return {};
},
R"pbdoc(
:sig=(p: Presentation, t: datetime.timedelta) -> tuple[list[int], list[int]] | tuple[str, str] | None:
:only-document-once:
)pbdoc");
m.def(
"todd_coxeter_class_by_index",
[](ToddCoxeter<Word>& tc, size_t n) {
auto c = todd_coxeter::class_by_index(tc, n);
// is this ok, does c somehow get copied into the iterator?
return py::make_iterator(rx::begin(c), rx::end(c));
},
py::arg("tc"),
py::arg("n"),
R"pbdoc(
:sig=(tc: ToddCoxeter, n: int) -> Iterator[list[int] | str]:
:only-document-once:
Returns an iterator yielding every word ``list[int]`` or ``str`` in the
congruence class with given index.
This function returns an iterator yielding every word belonging to the
class with index *n* in the congruence represented by the :any:`ToddCoxeter`
instance *tc*. Calls to this function trigger a full enumeration of *tc*.
:param tc: the :any:`ToddCoxeter` instance.
:type tc: ToddCoxeter
:param n: the index of the class.
:type n: int
:returns: A iterator yielding the class with index *n*.
:rtype: Iterator[list[int] | str]
:raises LibsemigroupsError:
if *n* is greater than or equal to ``tc.number_of_classes()``.
)pbdoc");
m.def(
"todd_coxeter_class_of",
[](ToddCoxeter<Word>& tc, Word const& w) {
auto c = todd_coxeter::class_of(tc, w);
// is this ok, does c somehow get copied into the iterator?
return py::make_iterator(rx::begin(c), rx::end(c));
},
py::arg("tc"),
py::arg("w"),
R"pbdoc(
:sig=(tc: ToddCoxeter, w: list[int] | str) -> Iterator[list[int] | str]:
:only-document-once:
Returns an iterator yielding every word (of the same type as *w*) in
the congruence class of the given word *w*.
This function returns an iterator yielding every word in belonging to
the same class as the input word *w* in the congruence represented by the
:any:`ToddCoxeter` instance *tc*. Calls to this function trigger a
full enumeration of *tc*.
:param tc: the :any:`ToddCoxeter` instance.
:type tc: ToddCoxeter
:param w: the input word.
:type w: list[int] | str
:returns: An iterator yielding words in the class of the input word.
:rtype: Iterator[list[int] | str]
:raises LibsemigroupsError:
if any of the values in *w* is out of range, i.e. they do not belong to
``presentation().alphabet()`` and ``Presentation.throw_if_letter_not_in_alphabet``
raises.)pbdoc");
m.def("todd_coxeter_is_non_trivial",
&todd_coxeter::is_non_trivial,
py::arg("tc"),
py::arg("tries") = 10,
py::arg("try_for") = std::chrono::milliseconds(100),
py::arg("threshold") = 0.99,
R"pbdoc(
:sig=(tc: ToddCoxeter, tries: int, try_for: datetime.timedelta, threshold: float) -> tril:
:only-document-once:
Check if the congruence has more than one class.
Returns :any:`tril.true` if it is possible to show that the congruence
is non-trivial; :any:`tril.false` if the congruence is already known to
be trivial; and :any:`tril.unknown` if it is not possible to show that
the congruence is non-trivial. This function attempts to find a
non-trivial congruence containing the congruence represented by a
:any:`ToddCoxeter` instance.
:param tc: the :any:`ToddCoxeter` instance.
:type tc: ToddCoxeter
:param tries:
the number of attempts to find a non-trivial super-congruence
(default: ``10``).
:type tries: int
:param try_for:
the amount of time to enumerate the congruence after choosing a random pair
of representatives and identifying them (default: 100 milliseconds).
:type try_for: datetime.timedelta
:param threshold: the threshold (default: ``0.99``).
:type threshold: float
:returns: Whether or not a non-trivial quotient was found.
:rtype: tril
)pbdoc");
m.def("todd_coxeter_perform_lookbehind",
&todd_coxeter::perform_lookbehind,
py::arg("tc"),
R"pbdoc(
:sig=(tc: ToddCoxeter) -> None:
:only-document-once:
Perform a lookbehind.
This function performs a "lookbehind" on the argument *tc* which is
defined as follows. For every node ``n`` in the so-far computed
:any:`WordGraph` (obtained from :any:`ToddCoxeter.current_word_graph`) we
use the current word graph to rewrite the current short-lex least path
from the initial node to ``n``. If this rewritten word is not equal to
the original word, and it also labels a path from the initial node in
the current word graph to a node ``m``, then ``m`` and ``n`` represent the
same congruence class. Thus we may collapse ``m`` and ``n`` (i.e. quotient
the word graph by the least congruence containing the pair ``m`` and
``n``).
The intended use case for this function is when you have a large word
graph in a partially enumerated :any:`ToddCoxeter` instance, and you
would like to minimise this word graph as far as possible.
For example, if we take the following monoid presentation of B. H.
Neumann for the trivial group:
.. code-block:: python
p = Presentation("abcdef")
p.contains_empty_word(true)
presentation.add_inverse_rules(p, "defabc")
presentation.add_rule(p, "bbdeaecbffdbaeeccefbccefb", "")
presentation.add_rule(p, "ccefbfacddecbffaafdcaafdc", "")
presentation.add_rule(p, "aafdcdbaeefacddbbdeabbdea", "")
tc = ToddCoxeter(congruence_kind.twosided, p)
Then running *tc* will simply grow the underlying word graph until
your computer runs out of memory. The authors of ``libsemigroups`` were
not able to find any combination of the many settings for
:any:`ToddCoxeter` where running *tc* returned an answer. We also tried
with GAP and ACE but neither of these seemed able to return an answer
either. But doing the following:
.. code-block:: python
tc.lookahead_extent(options.lookahead_extent.full)
.lookahead_style(options.lookahead_style.felsch)
tc.run_for(timedelta(seconds=1))
tc.perform_lookahead(True)
todd_coxeter.perform_lookbehind(tc)
tc.run_for(timedelta(seconds=1))
todd_coxeter.perform_lookbehind(tc)
tc.perform_lookahead(True)
tc.number_of_classes() # returns 1
returns the correct answer in about 22 seconds (on a 2024 Macbook Pro M4
Pro).
:param tc: the :any:`ToddCoxeter` instance.
:type tc: ToddCoxeter)pbdoc");
} // bind_todd_coxeter
} // namespace
void init_todd_coxeter(py::module& m) {
bind_todd_coxeter<word_type>(m, "ToddCoxeterWord");
bind_todd_coxeter<std::string>(m, "ToddCoxeterString");
}
} // namespace libsemigroups