snappy 3.2__cp312-cp312-macosx_10_13_x86_64.whl

Sign up to get free protection for your applications and to get access to all the features.
Files changed (503) hide show
  1. snappy/CyOpenGL.cpython-312-darwin.so +0 -0
  2. snappy/SnapPy.cpython-312-darwin.so +0 -0
  3. snappy/SnapPy.ico +0 -0
  4. snappy/SnapPy.png +0 -0
  5. snappy/SnapPyHP.cpython-312-darwin.so +0 -0
  6. snappy/__init__.py +760 -0
  7. snappy/app.py +605 -0
  8. snappy/app_menus.py +372 -0
  9. snappy/browser.py +998 -0
  10. snappy/cache.py +25 -0
  11. snappy/canonical.py +249 -0
  12. snappy/cusps/__init__.py +38 -0
  13. snappy/cusps/cusp_area_matrix.py +101 -0
  14. snappy/cusps/cusp_areas_from_matrix.py +173 -0
  15. snappy/cusps/maximal_cusp_area_matrix.py +136 -0
  16. snappy/cusps/test.py +21 -0
  17. snappy/cusps/trig_cusp_area_matrix.py +63 -0
  18. snappy/database.py +454 -0
  19. snappy/db_utilities.py +79 -0
  20. snappy/decorated_isosig.py +710 -0
  21. snappy/dev/__init__.py +0 -0
  22. snappy/dev/extended_ptolemy/__init__.py +8 -0
  23. snappy/dev/extended_ptolemy/closed.py +106 -0
  24. snappy/dev/extended_ptolemy/complexVolumesClosed.py +149 -0
  25. snappy/dev/extended_ptolemy/direct.py +42 -0
  26. snappy/dev/extended_ptolemy/extended.py +406 -0
  27. snappy/dev/extended_ptolemy/giac_helper.py +43 -0
  28. snappy/dev/extended_ptolemy/giac_rur.py +129 -0
  29. snappy/dev/extended_ptolemy/gluing.py +46 -0
  30. snappy/dev/extended_ptolemy/phc_wrapper.py +220 -0
  31. snappy/dev/extended_ptolemy/printMatrices.py +70 -0
  32. snappy/dev/vericlosed/__init__.py +1 -0
  33. snappy/dev/vericlosed/computeApproxHyperbolicStructureNew.py +159 -0
  34. snappy/dev/vericlosed/computeApproxHyperbolicStructureOrb.py +90 -0
  35. snappy/dev/vericlosed/computeVerifiedHyperbolicStructure.py +111 -0
  36. snappy/dev/vericlosed/gimbalLoopFinder.py +130 -0
  37. snappy/dev/vericlosed/hyperbolicStructure.py +313 -0
  38. snappy/dev/vericlosed/krawczykCertifiedEdgeLengthsEngine.py +165 -0
  39. snappy/dev/vericlosed/oneVertexTruncatedComplex.py +122 -0
  40. snappy/dev/vericlosed/orb/__init__.py +1 -0
  41. snappy/dev/vericlosed/orb/orb_solution_for_snappea_finite_triangulation_mac +0 -0
  42. snappy/dev/vericlosed/parseVertexGramMatrixFile.py +47 -0
  43. snappy/dev/vericlosed/polishApproxHyperbolicStructure.py +61 -0
  44. snappy/dev/vericlosed/test.py +54 -0
  45. snappy/dev/vericlosed/truncatedComplex.py +176 -0
  46. snappy/dev/vericlosed/verificationError.py +58 -0
  47. snappy/dev/vericlosed/verifyHyperbolicStructureEngine.py +177 -0
  48. snappy/doc/_images/SnapPy-196.png +0 -0
  49. snappy/doc/_images/geodesics.jpg +0 -0
  50. snappy/doc/_images/m004_paper_plane_on_systole.jpg +0 -0
  51. snappy/doc/_images/m125_paper_plane.jpg +0 -0
  52. snappy/doc/_images/mac.png +0 -0
  53. snappy/doc/_images/o9_00000_systole_paper_plane.jpg +0 -0
  54. snappy/doc/_images/o9_00000_systole_paper_plane_closer.jpg +0 -0
  55. snappy/doc/_images/plink-action.png +0 -0
  56. snappy/doc/_images/ubuntu.png +0 -0
  57. snappy/doc/_images/win7.png +0 -0
  58. snappy/doc/_sources/additional_classes.rst.txt +40 -0
  59. snappy/doc/_sources/bugs.rst.txt +14 -0
  60. snappy/doc/_sources/censuses.rst.txt +51 -0
  61. snappy/doc/_sources/credits.rst.txt +75 -0
  62. snappy/doc/_sources/development.rst.txt +259 -0
  63. snappy/doc/_sources/index.rst.txt +182 -0
  64. snappy/doc/_sources/installing.rst.txt +247 -0
  65. snappy/doc/_sources/manifold.rst.txt +6 -0
  66. snappy/doc/_sources/manifoldhp.rst.txt +46 -0
  67. snappy/doc/_sources/news.rst.txt +355 -0
  68. snappy/doc/_sources/other.rst.txt +25 -0
  69. snappy/doc/_sources/platonic_census.rst.txt +20 -0
  70. snappy/doc/_sources/plink.rst.txt +102 -0
  71. snappy/doc/_sources/ptolemy.rst.txt +66 -0
  72. snappy/doc/_sources/ptolemy_classes.rst.txt +42 -0
  73. snappy/doc/_sources/ptolemy_examples1.rst.txt +298 -0
  74. snappy/doc/_sources/ptolemy_examples2.rst.txt +363 -0
  75. snappy/doc/_sources/ptolemy_examples3.rst.txt +301 -0
  76. snappy/doc/_sources/ptolemy_examples4.rst.txt +61 -0
  77. snappy/doc/_sources/ptolemy_prelim.rst.txt +105 -0
  78. snappy/doc/_sources/screenshots.rst.txt +21 -0
  79. snappy/doc/_sources/snap.rst.txt +87 -0
  80. snappy/doc/_sources/snappy.rst.txt +28 -0
  81. snappy/doc/_sources/spherogram.rst.txt +103 -0
  82. snappy/doc/_sources/todo.rst.txt +47 -0
  83. snappy/doc/_sources/triangulation.rst.txt +11 -0
  84. snappy/doc/_sources/tutorial.rst.txt +49 -0
  85. snappy/doc/_sources/verify.rst.txt +210 -0
  86. snappy/doc/_sources/verify_internals.rst.txt +79 -0
  87. snappy/doc/_static/SnapPy-horizontal-128.png +0 -0
  88. snappy/doc/_static/SnapPy.ico +0 -0
  89. snappy/doc/_static/_sphinx_javascript_frameworks_compat.js +123 -0
  90. snappy/doc/_static/basic.css +925 -0
  91. snappy/doc/_static/css/badge_only.css +1 -0
  92. snappy/doc/_static/css/fonts/Roboto-Slab-Bold.woff +0 -0
  93. snappy/doc/_static/css/fonts/Roboto-Slab-Bold.woff2 +0 -0
  94. snappy/doc/_static/css/fonts/Roboto-Slab-Regular.woff +0 -0
  95. snappy/doc/_static/css/fonts/Roboto-Slab-Regular.woff2 +0 -0
  96. snappy/doc/_static/css/fonts/fontawesome-webfont.eot +0 -0
  97. snappy/doc/_static/css/fonts/fontawesome-webfont.svg +2671 -0
  98. snappy/doc/_static/css/fonts/fontawesome-webfont.ttf +0 -0
  99. snappy/doc/_static/css/fonts/fontawesome-webfont.woff +0 -0
  100. snappy/doc/_static/css/fonts/fontawesome-webfont.woff2 +0 -0
  101. snappy/doc/_static/css/fonts/lato-bold-italic.woff +0 -0
  102. snappy/doc/_static/css/fonts/lato-bold-italic.woff2 +0 -0
  103. snappy/doc/_static/css/fonts/lato-bold.woff +0 -0
  104. snappy/doc/_static/css/fonts/lato-bold.woff2 +0 -0
  105. snappy/doc/_static/css/fonts/lato-normal-italic.woff +0 -0
  106. snappy/doc/_static/css/fonts/lato-normal-italic.woff2 +0 -0
  107. snappy/doc/_static/css/fonts/lato-normal.woff +0 -0
  108. snappy/doc/_static/css/fonts/lato-normal.woff2 +0 -0
  109. snappy/doc/_static/css/theme.css +4 -0
  110. snappy/doc/_static/doctools.js +156 -0
  111. snappy/doc/_static/documentation_options.js +13 -0
  112. snappy/doc/_static/file.png +0 -0
  113. snappy/doc/_static/fonts/Lato/lato-bold.eot +0 -0
  114. snappy/doc/_static/fonts/Lato/lato-bold.ttf +0 -0
  115. snappy/doc/_static/fonts/Lato/lato-bold.woff +0 -0
  116. snappy/doc/_static/fonts/Lato/lato-bold.woff2 +0 -0
  117. snappy/doc/_static/fonts/Lato/lato-bolditalic.eot +0 -0
  118. snappy/doc/_static/fonts/Lato/lato-bolditalic.ttf +0 -0
  119. snappy/doc/_static/fonts/Lato/lato-bolditalic.woff +0 -0
  120. snappy/doc/_static/fonts/Lato/lato-bolditalic.woff2 +0 -0
  121. snappy/doc/_static/fonts/Lato/lato-italic.eot +0 -0
  122. snappy/doc/_static/fonts/Lato/lato-italic.ttf +0 -0
  123. snappy/doc/_static/fonts/Lato/lato-italic.woff +0 -0
  124. snappy/doc/_static/fonts/Lato/lato-italic.woff2 +0 -0
  125. snappy/doc/_static/fonts/Lato/lato-regular.eot +0 -0
  126. snappy/doc/_static/fonts/Lato/lato-regular.ttf +0 -0
  127. snappy/doc/_static/fonts/Lato/lato-regular.woff +0 -0
  128. snappy/doc/_static/fonts/Lato/lato-regular.woff2 +0 -0
  129. snappy/doc/_static/fonts/RobotoSlab/roboto-slab-v7-bold.eot +0 -0
  130. snappy/doc/_static/fonts/RobotoSlab/roboto-slab-v7-bold.ttf +0 -0
  131. snappy/doc/_static/fonts/RobotoSlab/roboto-slab-v7-bold.woff +0 -0
  132. snappy/doc/_static/fonts/RobotoSlab/roboto-slab-v7-bold.woff2 +0 -0
  133. snappy/doc/_static/fonts/RobotoSlab/roboto-slab-v7-regular.eot +0 -0
  134. snappy/doc/_static/fonts/RobotoSlab/roboto-slab-v7-regular.ttf +0 -0
  135. snappy/doc/_static/fonts/RobotoSlab/roboto-slab-v7-regular.woff +0 -0
  136. snappy/doc/_static/fonts/RobotoSlab/roboto-slab-v7-regular.woff2 +0 -0
  137. snappy/doc/_static/jquery.js +2 -0
  138. snappy/doc/_static/js/badge_only.js +1 -0
  139. snappy/doc/_static/js/theme.js +1 -0
  140. snappy/doc/_static/js/versions.js +228 -0
  141. snappy/doc/_static/language_data.js +199 -0
  142. snappy/doc/_static/minus.png +0 -0
  143. snappy/doc/_static/plus.png +0 -0
  144. snappy/doc/_static/pygments.css +75 -0
  145. snappy/doc/_static/searchtools.js +620 -0
  146. snappy/doc/_static/snappy_furo.css +33 -0
  147. snappy/doc/_static/snappy_sphinx_rtd_theme.css +42 -0
  148. snappy/doc/_static/sphinx_highlight.js +154 -0
  149. snappy/doc/additional_classes.html +1500 -0
  150. snappy/doc/bugs.html +132 -0
  151. snappy/doc/censuses.html +427 -0
  152. snappy/doc/credits.html +181 -0
  153. snappy/doc/development.html +384 -0
  154. snappy/doc/genindex.html +1331 -0
  155. snappy/doc/index.html +262 -0
  156. snappy/doc/installing.html +346 -0
  157. snappy/doc/manifold.html +3452 -0
  158. snappy/doc/manifoldhp.html +180 -0
  159. snappy/doc/news.html +388 -0
  160. snappy/doc/objects.inv +0 -0
  161. snappy/doc/other.html +161 -0
  162. snappy/doc/platonic_census.html +375 -0
  163. snappy/doc/plink.html +210 -0
  164. snappy/doc/ptolemy.html +254 -0
  165. snappy/doc/ptolemy_classes.html +1144 -0
  166. snappy/doc/ptolemy_examples1.html +409 -0
  167. snappy/doc/ptolemy_examples2.html +471 -0
  168. snappy/doc/ptolemy_examples3.html +414 -0
  169. snappy/doc/ptolemy_examples4.html +195 -0
  170. snappy/doc/ptolemy_prelim.html +248 -0
  171. snappy/doc/py-modindex.html +165 -0
  172. snappy/doc/screenshots.html +141 -0
  173. snappy/doc/search.html +135 -0
  174. snappy/doc/searchindex.js +1 -0
  175. snappy/doc/snap.html +202 -0
  176. snappy/doc/snappy.html +181 -0
  177. snappy/doc/spherogram.html +1211 -0
  178. snappy/doc/todo.html +166 -0
  179. snappy/doc/triangulation.html +1584 -0
  180. snappy/doc/tutorial.html +159 -0
  181. snappy/doc/verify.html +330 -0
  182. snappy/doc/verify_internals.html +1235 -0
  183. snappy/drilling/__init__.py +456 -0
  184. snappy/drilling/barycentric.py +103 -0
  185. snappy/drilling/constants.py +5 -0
  186. snappy/drilling/crush.py +270 -0
  187. snappy/drilling/cusps.py +125 -0
  188. snappy/drilling/debug.py +242 -0
  189. snappy/drilling/epsilons.py +6 -0
  190. snappy/drilling/exceptions.py +55 -0
  191. snappy/drilling/moves.py +620 -0
  192. snappy/drilling/peripheral_curves.py +210 -0
  193. snappy/drilling/perturb.py +188 -0
  194. snappy/drilling/shorten.py +36 -0
  195. snappy/drilling/subdivide.py +274 -0
  196. snappy/drilling/test.py +23 -0
  197. snappy/drilling/test_cases.py +126 -0
  198. snappy/drilling/tracing.py +351 -0
  199. snappy/exceptions.py +26 -0
  200. snappy/export_stl.py +120 -0
  201. snappy/exterior_to_link/__init__.py +2 -0
  202. snappy/exterior_to_link/barycentric_geometry.py +463 -0
  203. snappy/exterior_to_link/exceptions.py +6 -0
  204. snappy/exterior_to_link/geodesic_map.json +14408 -0
  205. snappy/exterior_to_link/hyp_utils.py +112 -0
  206. snappy/exterior_to_link/link_projection.py +323 -0
  207. snappy/exterior_to_link/main.py +197 -0
  208. snappy/exterior_to_link/mcomplex_with_expansion.py +261 -0
  209. snappy/exterior_to_link/mcomplex_with_link.py +687 -0
  210. snappy/exterior_to_link/mcomplex_with_memory.py +162 -0
  211. snappy/exterior_to_link/pl_utils.py +491 -0
  212. snappy/exterior_to_link/put_in_S3.py +156 -0
  213. snappy/exterior_to_link/rational_linear_algebra.py +123 -0
  214. snappy/exterior_to_link/rational_linear_algebra_wrapped.py +135 -0
  215. snappy/exterior_to_link/simplify_to_base_tri.py +114 -0
  216. snappy/exterior_to_link/stored_moves.py +475 -0
  217. snappy/exterior_to_link/test.py +31 -0
  218. snappy/filedialog.py +28 -0
  219. snappy/geometric_structure/__init__.py +212 -0
  220. snappy/geometric_structure/cusp_neighborhood/__init__.py +3 -0
  221. snappy/geometric_structure/cusp_neighborhood/complex_cusp_cross_section.py +697 -0
  222. snappy/geometric_structure/cusp_neighborhood/cusp_cross_section_base.py +484 -0
  223. snappy/geometric_structure/cusp_neighborhood/exceptions.py +42 -0
  224. snappy/geometric_structure/cusp_neighborhood/real_cusp_cross_section.py +298 -0
  225. snappy/geometric_structure/cusp_neighborhood/tiles_for_cusp_neighborhood.py +159 -0
  226. snappy/geometric_structure/cusp_neighborhood/vertices.py +32 -0
  227. snappy/geometric_structure/geodesic/__init__.py +0 -0
  228. snappy/geometric_structure/geodesic/add_core_curves.py +152 -0
  229. snappy/geometric_structure/geodesic/avoid_core_curves.py +369 -0
  230. snappy/geometric_structure/geodesic/canonical_keys.py +52 -0
  231. snappy/geometric_structure/geodesic/check_away_from_core_curve.py +60 -0
  232. snappy/geometric_structure/geodesic/constants.py +6 -0
  233. snappy/geometric_structure/geodesic/exceptions.py +22 -0
  234. snappy/geometric_structure/geodesic/fixed_points.py +93 -0
  235. snappy/geometric_structure/geodesic/geodesic_start_point_info.py +435 -0
  236. snappy/geometric_structure/geodesic/graph_trace_helper.py +67 -0
  237. snappy/geometric_structure/geodesic/line.py +30 -0
  238. snappy/geometric_structure/geodesic/multiplicity.py +127 -0
  239. snappy/geometric_structure/geodesic/tiles_for_geodesic.py +101 -0
  240. snappy/geometric_structure/test.py +22 -0
  241. snappy/gui.py +121 -0
  242. snappy/horoviewer.py +443 -0
  243. snappy/hyperboloid/__init__.py +212 -0
  244. snappy/hyperboloid/distances.py +245 -0
  245. snappy/hyperboloid/horoball.py +19 -0
  246. snappy/hyperboloid/line.py +35 -0
  247. snappy/hyperboloid/point.py +9 -0
  248. snappy/hyperboloid/triangle.py +29 -0
  249. snappy/info_icon.gif +0 -0
  250. snappy/infowindow.py +65 -0
  251. snappy/isometry_signature.py +382 -0
  252. snappy/len_spec/__init__.py +596 -0
  253. snappy/len_spec/geodesic_info.py +110 -0
  254. snappy/len_spec/geodesic_key_info_dict.py +117 -0
  255. snappy/len_spec/geodesic_piece.py +143 -0
  256. snappy/len_spec/geometric_structure.py +182 -0
  257. snappy/len_spec/geometry.py +80 -0
  258. snappy/len_spec/length_spectrum_geodesic_info.py +170 -0
  259. snappy/len_spec/spine.py +206 -0
  260. snappy/len_spec/test.py +24 -0
  261. snappy/len_spec/test_cases.py +69 -0
  262. snappy/len_spec/tile.py +275 -0
  263. snappy/len_spec/word.py +86 -0
  264. snappy/manifolds/HTWKnots/alternating.gz +0 -0
  265. snappy/manifolds/HTWKnots/nonalternating.gz +0 -0
  266. snappy/manifolds/__init__.py +3 -0
  267. snappy/math_basics.py +176 -0
  268. snappy/matrix.py +525 -0
  269. snappy/number.py +657 -0
  270. snappy/numeric_output_checker.py +345 -0
  271. snappy/pari.py +41 -0
  272. snappy/phone_home.py +57 -0
  273. snappy/polyviewer.py +259 -0
  274. snappy/ptolemy/__init__.py +17 -0
  275. snappy/ptolemy/component.py +103 -0
  276. snappy/ptolemy/coordinates.py +2290 -0
  277. snappy/ptolemy/fieldExtensions.py +153 -0
  278. snappy/ptolemy/findLoops.py +473 -0
  279. snappy/ptolemy/geometricRep.py +59 -0
  280. snappy/ptolemy/homology.py +165 -0
  281. snappy/ptolemy/magma/default.magma_template +229 -0
  282. snappy/ptolemy/magma/radicalsOfPrimaryDecomposition.magma_template +79 -0
  283. snappy/ptolemy/manifoldMethods.py +395 -0
  284. snappy/ptolemy/matrix.py +350 -0
  285. snappy/ptolemy/numericalSolutionsToGroebnerBasis.py +113 -0
  286. snappy/ptolemy/polynomial.py +857 -0
  287. snappy/ptolemy/processComponents.py +173 -0
  288. snappy/ptolemy/processFileBase.py +247 -0
  289. snappy/ptolemy/processFileDispatch.py +46 -0
  290. snappy/ptolemy/processMagmaFile.py +392 -0
  291. snappy/ptolemy/processRurFile.py +150 -0
  292. snappy/ptolemy/ptolemyGeneralizedObstructionClass.py +102 -0
  293. snappy/ptolemy/ptolemyObstructionClass.py +64 -0
  294. snappy/ptolemy/ptolemyVariety.py +1029 -0
  295. snappy/ptolemy/ptolemyVarietyPrimeIdealGroebnerBasis.py +140 -0
  296. snappy/ptolemy/reginaWrapper.py +698 -0
  297. snappy/ptolemy/regina_testing_files/DT_mcbbiceaibjklmdfgh__sl2_c0.magma_out.bz2 +0 -0
  298. snappy/ptolemy/regina_testing_files/DT_mcbbiceaibjklmdfgh__sl2_c1.magma_out.bz2 +0 -0
  299. snappy/ptolemy/regina_testing_files/DT_mcbbiceaibjklmdfgh__sl2_c2.magma_out.bz2 +0 -0
  300. snappy/ptolemy/regina_testing_files/DT_mcbbiceaibjklmdfgh__sl2_c3.magma_out.bz2 +0 -0
  301. snappy/ptolemy/regina_testing_files/DT_mcbbiceaibjklmdfgh__sl2_c4.magma_out.bz2 +0 -0
  302. snappy/ptolemy/regina_testing_files/DT_mcbbiceaibjklmdfgh__sl2_c5.magma_out.bz2 +0 -0
  303. snappy/ptolemy/regina_testing_files/DT_mcbbiceaibjklmdfgh__sl2_c6.magma_out.bz2 +0 -0
  304. snappy/ptolemy/regina_testing_files/DT_mcbbiceaibjklmdfgh__sl2_c7.magma_out.bz2 +0 -0
  305. snappy/ptolemy/regina_testing_files_generalized/m003__sl3_c0.magma_out.bz2 +0 -0
  306. snappy/ptolemy/regina_testing_files_generalized/m003__sl3_c1.magma_out.bz2 +0 -0
  307. snappy/ptolemy/regina_testing_files_generalized/m015__sl2_c0.magma_out.bz2 +0 -0
  308. snappy/ptolemy/regina_testing_files_generalized/m015__sl2_c1.magma_out.bz2 +0 -0
  309. snappy/ptolemy/regina_testing_files_generalized/m015__sl3_c0.magma_out.bz2 +0 -0
  310. snappy/ptolemy/regina_testing_files_generalized/m015__sl3_c1.magma_out.bz2 +0 -0
  311. snappy/ptolemy/rur.py +545 -0
  312. snappy/ptolemy/solutionsToPrimeIdealGroebnerBasis.py +277 -0
  313. snappy/ptolemy/test.py +1126 -0
  314. snappy/ptolemy/testing_files/3_1__sl2_c0.magma_out.bz2 +0 -0
  315. snappy/ptolemy/testing_files/3_1__sl2_c1.magma_out.bz2 +0 -0
  316. snappy/ptolemy/testing_files/4_1__sl2_c0.magma_out.bz2 +0 -0
  317. snappy/ptolemy/testing_files/4_1__sl2_c1.magma_out.bz2 +0 -0
  318. snappy/ptolemy/testing_files/4_1__sl3_c0.magma_out.bz2 +0 -0
  319. snappy/ptolemy/testing_files/4_1__sl4_c0.magma_out.bz2 +0 -0
  320. snappy/ptolemy/testing_files/4_1__sl4_c1.magma_out.bz2 +0 -0
  321. snappy/ptolemy/testing_files/5_2__sl2_c0.magma_out.bz2 +0 -0
  322. snappy/ptolemy/testing_files/5_2__sl2_c1.magma_out.bz2 +0 -0
  323. snappy/ptolemy/testing_files/DT_mcbbiceaibjklmdfgh__sl2_c0.magma_out.bz2 +0 -0
  324. snappy/ptolemy/testing_files/DT_mcbbiceaibjklmdfgh__sl2_c1.magma_out.bz2 +0 -0
  325. snappy/ptolemy/testing_files/DT_mcbbiceaibjklmdfgh__sl2_c2.magma_out.bz2 +0 -0
  326. snappy/ptolemy/testing_files/DT_mcbbiceaibjklmdfgh__sl2_c3.magma_out.bz2 +0 -0
  327. snappy/ptolemy/testing_files/DT_mcbbiceaibjklmdfgh__sl2_c4.magma_out.bz2 +0 -0
  328. snappy/ptolemy/testing_files/DT_mcbbiceaibjklmdfgh__sl2_c5.magma_out.bz2 +0 -0
  329. snappy/ptolemy/testing_files/DT_mcbbiceaibjklmdfgh__sl2_c6.magma_out.bz2 +0 -0
  330. snappy/ptolemy/testing_files/DT_mcbbiceaibjklmdfgh__sl2_c7.magma_out.bz2 +0 -0
  331. snappy/ptolemy/testing_files/data/pgl2/OrientableCuspedCensus/03_tetrahedra/m019__sl2_c0.magma_out +95 -0
  332. snappy/ptolemy/testing_files/data/pgl2/OrientableCuspedCensus/03_tetrahedra/m019__sl2_c1.magma_out +95 -0
  333. snappy/ptolemy/testing_files/m015__sl3_c0.magma_out.bz2 +0 -0
  334. snappy/ptolemy/testing_files/m135__sl2_c0.magma_out.bz2 +0 -0
  335. snappy/ptolemy/testing_files/m135__sl2_c1.magma_out.bz2 +0 -0
  336. snappy/ptolemy/testing_files/m135__sl2_c2.magma_out.bz2 +0 -0
  337. snappy/ptolemy/testing_files/m135__sl2_c3.magma_out.bz2 +0 -0
  338. snappy/ptolemy/testing_files/m135__sl2_c4.magma_out.bz2 +0 -0
  339. snappy/ptolemy/testing_files/m135__sl2_c5.magma_out.bz2 +0 -0
  340. snappy/ptolemy/testing_files/m135__sl2_c6.magma_out.bz2 +0 -0
  341. snappy/ptolemy/testing_files/m135__sl2_c7.magma_out.bz2 +0 -0
  342. snappy/ptolemy/testing_files/s000__sl2_c0.magma_out.bz2 +0 -0
  343. snappy/ptolemy/testing_files/s000__sl2_c1.magma_out.bz2 +0 -0
  344. snappy/ptolemy/testing_files/t00000__sl2_c0.magma_out.bz2 +0 -0
  345. snappy/ptolemy/testing_files/t00000__sl2_c1.magma_out.bz2 +0 -0
  346. snappy/ptolemy/testing_files/v0000__sl2_c0.magma_out.bz2 +0 -0
  347. snappy/ptolemy/testing_files/v0000__sl2_c1.magma_out.bz2 +0 -0
  348. snappy/ptolemy/testing_files/v0000__sl2_c2.magma_out.bz2 +0 -0
  349. snappy/ptolemy/testing_files/v0000__sl2_c3.magma_out.bz2 +0 -0
  350. snappy/ptolemy/testing_files_generalized/m003__sl2_c0.magma_out.bz2 +0 -0
  351. snappy/ptolemy/testing_files_generalized/m003__sl2_c1.magma_out.bz2 +0 -0
  352. snappy/ptolemy/testing_files_generalized/m003__sl3_c0.magma_out.bz2 +0 -0
  353. snappy/ptolemy/testing_files_generalized/m003__sl3_c1.magma_out.bz2 +0 -0
  354. snappy/ptolemy/testing_files_generalized/m004__sl2_c0.magma_out.bz2 +0 -0
  355. snappy/ptolemy/testing_files_generalized/m004__sl2_c1.magma_out.bz2 +0 -0
  356. snappy/ptolemy/testing_files_generalized/m015__sl2_c1.magma_out.bz2 +0 -0
  357. snappy/ptolemy/testing_files_generalized/m015__sl3_c0.magma_out.bz2 +0 -0
  358. snappy/ptolemy/testing_files_rur/m052__sl3_c0.rur.bz2 +0 -0
  359. snappy/ptolemy/utilities.py +236 -0
  360. snappy/raytracing/__init__.py +64 -0
  361. snappy/raytracing/additional_horospheres.py +64 -0
  362. snappy/raytracing/additional_len_spec_choices.py +63 -0
  363. snappy/raytracing/cohomology_fractal.py +197 -0
  364. snappy/raytracing/eyeball.py +123 -0
  365. snappy/raytracing/finite_raytracing_data.py +237 -0
  366. snappy/raytracing/finite_viewer.py +590 -0
  367. snappy/raytracing/geodesic_tube_info.py +174 -0
  368. snappy/raytracing/geodesics.py +246 -0
  369. snappy/raytracing/geodesics_window.py +258 -0
  370. snappy/raytracing/gui_utilities.py +293 -0
  371. snappy/raytracing/hyperboloid_navigation.py +556 -0
  372. snappy/raytracing/hyperboloid_utilities.py +234 -0
  373. snappy/raytracing/ideal_raytracing_data.py +592 -0
  374. snappy/raytracing/inside_viewer.py +974 -0
  375. snappy/raytracing/pack.py +22 -0
  376. snappy/raytracing/raytracing_data.py +126 -0
  377. snappy/raytracing/raytracing_view.py +454 -0
  378. snappy/raytracing/shaders/Eye.png +0 -0
  379. snappy/raytracing/shaders/NonGeometric.png +0 -0
  380. snappy/raytracing/shaders/__init__.py +101 -0
  381. snappy/raytracing/shaders/fragment.glsl +1744 -0
  382. snappy/raytracing/test.py +29 -0
  383. snappy/raytracing/tooltip.py +146 -0
  384. snappy/raytracing/upper_halfspace_utilities.py +98 -0
  385. snappy/raytracing/view_scale_controller.py +98 -0
  386. snappy/raytracing/zoom_slider/__init__.py +263 -0
  387. snappy/raytracing/zoom_slider/inward.png +0 -0
  388. snappy/raytracing/zoom_slider/inward18.png +0 -0
  389. snappy/raytracing/zoom_slider/outward.png +0 -0
  390. snappy/raytracing/zoom_slider/outward18.png +0 -0
  391. snappy/raytracing/zoom_slider/test.py +20 -0
  392. snappy/sage_helper.py +117 -0
  393. snappy/settings.py +409 -0
  394. snappy/shell.py +53 -0
  395. snappy/snap/__init__.py +114 -0
  396. snappy/snap/character_varieties.py +375 -0
  397. snappy/snap/find_field.py +372 -0
  398. snappy/snap/fundamental_polyhedron.py +569 -0
  399. snappy/snap/generators.py +39 -0
  400. snappy/snap/interval_reps.py +81 -0
  401. snappy/snap/kernel_structures.py +128 -0
  402. snappy/snap/mcomplex_base.py +18 -0
  403. snappy/snap/nsagetools.py +702 -0
  404. snappy/snap/peripheral/__init__.py +1 -0
  405. snappy/snap/peripheral/dual_cellulation.py +219 -0
  406. snappy/snap/peripheral/link.py +127 -0
  407. snappy/snap/peripheral/peripheral.py +159 -0
  408. snappy/snap/peripheral/surface.py +522 -0
  409. snappy/snap/peripheral/test.py +35 -0
  410. snappy/snap/polished_reps.py +335 -0
  411. snappy/snap/shapes.py +152 -0
  412. snappy/snap/slice_obs_HKL.py +668 -0
  413. snappy/snap/t3mlite/__init__.py +2 -0
  414. snappy/snap/t3mlite/arrow.py +243 -0
  415. snappy/snap/t3mlite/corner.py +22 -0
  416. snappy/snap/t3mlite/edge.py +172 -0
  417. snappy/snap/t3mlite/face.py +37 -0
  418. snappy/snap/t3mlite/files.py +211 -0
  419. snappy/snap/t3mlite/homology.py +53 -0
  420. snappy/snap/t3mlite/linalg.py +419 -0
  421. snappy/snap/t3mlite/mcomplex.py +1499 -0
  422. snappy/snap/t3mlite/perm4.py +320 -0
  423. snappy/snap/t3mlite/setup.py +12 -0
  424. snappy/snap/t3mlite/simplex.py +199 -0
  425. snappy/snap/t3mlite/spun.py +297 -0
  426. snappy/snap/t3mlite/surface.py +519 -0
  427. snappy/snap/t3mlite/test.py +20 -0
  428. snappy/snap/t3mlite/test_vs_regina.py +86 -0
  429. snappy/snap/t3mlite/tetrahedron.py +109 -0
  430. snappy/snap/t3mlite/vertex.py +42 -0
  431. snappy/snap/test.py +134 -0
  432. snappy/snap/utilities.py +288 -0
  433. snappy/test.py +209 -0
  434. snappy/test_cases.py +263 -0
  435. snappy/testing.py +131 -0
  436. snappy/tiling/__init__.py +2 -0
  437. snappy/tiling/canonical_key_dict.py +59 -0
  438. snappy/tiling/dict_based_set.py +79 -0
  439. snappy/tiling/floor.py +49 -0
  440. snappy/tiling/hyperboloid_dict.py +54 -0
  441. snappy/tiling/iter_utils.py +78 -0
  442. snappy/tiling/lifted_tetrahedron.py +22 -0
  443. snappy/tiling/lifted_tetrahedron_set.py +54 -0
  444. snappy/tiling/real_hash_dict.py +164 -0
  445. snappy/tiling/test.py +23 -0
  446. snappy/tiling/tile.py +215 -0
  447. snappy/tiling/triangle.py +33 -0
  448. snappy/tkterminal.py +920 -0
  449. snappy/twister/__init__.py +20 -0
  450. snappy/twister/main.py +646 -0
  451. snappy/twister/surfaces/S_0_1 +3 -0
  452. snappy/twister/surfaces/S_0_2 +3 -0
  453. snappy/twister/surfaces/S_0_4 +7 -0
  454. snappy/twister/surfaces/S_0_4_Lantern +8 -0
  455. snappy/twister/surfaces/S_1 +3 -0
  456. snappy/twister/surfaces/S_1_1 +4 -0
  457. snappy/twister/surfaces/S_1_2 +5 -0
  458. snappy/twister/surfaces/S_1_2_5 +6 -0
  459. snappy/twister/surfaces/S_2 +6 -0
  460. snappy/twister/surfaces/S_2_1 +8 -0
  461. snappy/twister/surfaces/S_2_heeg +10 -0
  462. snappy/twister/surfaces/S_3 +8 -0
  463. snappy/twister/surfaces/S_3_1 +10 -0
  464. snappy/twister/surfaces/S_4_1 +12 -0
  465. snappy/twister/surfaces/S_5_1 +14 -0
  466. snappy/twister/surfaces/heeg_fig8 +9 -0
  467. snappy/twister/twister_core.cpython-312-darwin.so +0 -0
  468. snappy/upper_halfspace/__init__.py +146 -0
  469. snappy/upper_halfspace/ideal_point.py +26 -0
  470. snappy/verify/__init__.py +13 -0
  471. snappy/verify/canonical.py +542 -0
  472. snappy/verify/complex_volume/__init__.py +18 -0
  473. snappy/verify/complex_volume/adjust_torsion.py +86 -0
  474. snappy/verify/complex_volume/closed.py +168 -0
  475. snappy/verify/complex_volume/compute_ptolemys.py +90 -0
  476. snappy/verify/complex_volume/cusped.py +56 -0
  477. snappy/verify/complex_volume/extended_bloch.py +201 -0
  478. snappy/verify/cusp_translations.py +85 -0
  479. snappy/verify/edge_equations.py +80 -0
  480. snappy/verify/exceptions.py +254 -0
  481. snappy/verify/hyperbolicity.py +224 -0
  482. snappy/verify/interval_newton_shapes_engine.py +523 -0
  483. snappy/verify/interval_tree.py +400 -0
  484. snappy/verify/krawczyk_shapes_engine.py +518 -0
  485. snappy/verify/maximal_cusp_area_matrix/__init__.py +46 -0
  486. snappy/verify/maximal_cusp_area_matrix/cusp_tiling_engine.py +419 -0
  487. snappy/verify/maximal_cusp_area_matrix/cusp_translate_engine.py +153 -0
  488. snappy/verify/real_algebra.py +286 -0
  489. snappy/verify/shapes.py +25 -0
  490. snappy/verify/short_slopes.py +200 -0
  491. snappy/verify/square_extensions.py +1005 -0
  492. snappy/verify/test.py +78 -0
  493. snappy/verify/upper_halfspace/__init__.py +9 -0
  494. snappy/verify/upper_halfspace/extended_matrix.py +100 -0
  495. snappy/verify/upper_halfspace/finite_point.py +283 -0
  496. snappy/verify/upper_halfspace/ideal_point.py +426 -0
  497. snappy/verify/volume.py +128 -0
  498. snappy/version.py +2 -0
  499. snappy-3.2.dist-info/METADATA +58 -0
  500. snappy-3.2.dist-info/RECORD +503 -0
  501. snappy-3.2.dist-info/WHEEL +5 -0
  502. snappy-3.2.dist-info/entry_points.txt +2 -0
  503. snappy-3.2.dist-info/top_level.txt +28 -0
@@ -0,0 +1,596 @@
1
+ from .geometric_structure import mcomplex_for_len_spec
2
+ from .tile import compute_length_spectrum_tiles, LengthSpectrumTile
3
+ from .geodesic_info import GeodesicInfoBase, GeodesicKeyInfo, CoreCurveGeodesicInfo
4
+ from .geodesic_key_info_dict import get_geodesic_key_info_set
5
+ from .word import simplify_geodesic_word
6
+ from .length_spectrum_geodesic_info import LengthSpectrumGeodesicInfo
7
+
8
+ from ..snap.t3mlite import Mcomplex
9
+ from ..geometric_structure import word_list_to_psl2c_matrix
10
+ from ..upper_halfspace import psl2c_to_o13
11
+ from ..hyperboloid.distances import distance_r13_point_line
12
+ from ..hyperboloid.line import R13Line
13
+ from ..SnapPy import list_as_word, inverse_list_word
14
+ from ..math_basics import lower, correct_max # type: ignore
15
+ from ..exceptions import InsufficientPrecisionError, NonorientableManifoldError
16
+ from ..sage_helper import _within_sage, SageNotAvailable
17
+
18
+ if _within_sage:
19
+ from sage.rings.real_mpfi import RealIntervalField
20
+
21
+ import heapq
22
+
23
+ from typing import Any, Optional, List, Sequence
24
+
25
+ _optimization : bool = False
26
+
27
+ def length_spectrum_alt_gen(manifold,
28
+ bits_prec : Optional[int] = None,
29
+ verified : bool = False
30
+ ) -> Sequence[LengthSpectrumGeodesicInfo]:
31
+ """
32
+ Returns a generator for the geodesics sorted by real length. The method
33
+ only supports orientable manifolds.
34
+
35
+ Here is an example::
36
+
37
+ >>> M = Manifold("m202(3,4)(0,0)")
38
+ >>> spec = M.length_spectrum_alt_gen()
39
+ >>> next(spec) # doctest: +NUMERIC9
40
+ Length Core curve Word
41
+ 0.14742465268512 - 1.78287093565202*I Cusp 0 aabcDabcB
42
+ >>> next(spec) # doctest: +NUMERIC9
43
+ 0.81161414965958 + 2.72911699294426*I - b
44
+ >>> next(spec) # doctest: +NUMERIC9
45
+ 0.84163270359334 + 2.61245944742151*I - aB
46
+
47
+ Note that the shortest geodesic in the above example happens to be the
48
+ core curve of the filled cusp (Cusp 0).
49
+
50
+ Access just the length or word::
51
+
52
+ >>> g = next(spec)
53
+ >>> g.length # doctest: +NUMERIC9
54
+ 0.93461379591349 + 2.70060614107722*I
55
+ >>> g.word
56
+ 'a'
57
+
58
+ The word is given with respect to the unsimplified fundamental group::
59
+
60
+ >>> G = M.fundamental_group(simplify_presentation=False)
61
+ >>> G.complex_length('a') # doctest: +NUMERIC9
62
+ 0.93461379591349 + 2.70060614107722*I
63
+
64
+ The method also supports higher precision::
65
+
66
+ >>> M = Manifold("m003(-3,1)")
67
+ >>> spec = M.length_spectrum_alt_gen(bits_prec=100)
68
+ >>> next(spec).length # doctest: +NUMERIC24
69
+ 0.58460368501798696932015666264 + 2.4953704555604684110903962008*I
70
+
71
+ **Performance**
72
+
73
+ This method uses a different algorithm than
74
+ :meth:`length_spectrum <Manifold.length_spectrum>`. In particular,
75
+ it does not compute the Dirichlet domain. It also allows for
76
+ :ref:`verified computations <verify-primer>`.
77
+ It is implemented in python and thus
78
+ typically slower than :meth:`length_spectrum <Manifold.length_spectrum>`.
79
+ But there are also some cases where it is significantly faster. In
80
+ particular, this applies to spun triangulations such as ``m004(21,10)``.
81
+
82
+ Here is example where we can help the algorithm by guessing and drilling
83
+ and filling a short geodesic::
84
+
85
+ >>> M = Manifold("o9_00639")
86
+
87
+ Over an hour to compute::
88
+
89
+ >>> M.high_precision().length_spectrum(0.1) # doctest: +SKIP
90
+ mult length topology parity
91
+ 1 0.00150226276052 - 2.39996262244127*I circle +
92
+
93
+ A couple of minutes to compute::
94
+
95
+ >>> spec = M.length_spectrum_alt_gen(bits_prec = 150)
96
+ >>> next(spec) # doctest: +SKIP
97
+ Length Word Core curve
98
+ 0.00150226276052 - 2.39996262244127*I a -
99
+
100
+ After drilling and filling, less than a second to compute::
101
+
102
+ >>> N = M.drill_word('a')
103
+ >>> N.dehn_fill((1,0),-1) # N is now isometric to o9_00639 but as a surgery m125(0,0)(34,55)
104
+ >>> spec = N.length_spectrum_alt_gen()
105
+ >>> next(spec) # doctest: +NUMERIC9
106
+ Length Core curve Word
107
+ 0.00150226276073 - 2.39996262244128*I Cusp 1 cDcDDcDcDDcDDcDcDDcDcDDcDDcDcDDcDD
108
+ >>> next(spec).length.real() # doctest: +NUMERIC9
109
+ 0.96218768626877
110
+
111
+ **Verified computations**
112
+
113
+ The method also supports :ref:`verified computations <verify-primer>`::
114
+
115
+ sage: M = Manifold("m019")
116
+ sage: spec = M.length_spectrum_alt_gen(verified=True, bits_prec=100)
117
+ sage: next(spec)
118
+ Length Core curve Word
119
+ 0.43153441294719... + 2.35105908147863...*I - a
120
+ sage: next(spec)
121
+ 0.88944299721255... - 2.94185904702273...*I - bD
122
+
123
+ If :attr:`verified = True` is passed, the algorithm guarantees that the lower
124
+ bound of the real length is (non-strictly) increasing. In particular, we know
125
+ that we have found all geodesics less than the following length::
126
+
127
+ sage: next(spec).length.real().lower() # doctest: +NUMERIC12
128
+ 0.94135129037387168886341739832
129
+
130
+ To illustrate some pitfalls, here is an example of a potential a result
131
+ of the method:
132
+
133
+ +----------------------+-------+
134
+ | Real length interval | Word |
135
+ +======================+=======+
136
+ | ``[1.0, 2.0]`` | ``a`` |
137
+ +----------------------+-------+
138
+ | ``[1.2, 1.3]`` | ``b`` |
139
+ +----------------------+-------+
140
+ | ``[1.7, 1.8]`` | ``c`` |
141
+ +----------------------+-------+
142
+ | ``[3.0, 4.0]`` | ``d`` |
143
+ +----------------------+-------+
144
+
145
+ Note that we cannot say whether geodesic ``a`` is actually the first,
146
+ second or third shortest geodesic or tied with ``b`` or ``c``. Increasing
147
+ precision can change (representative words and) the order in which the
148
+ geodesics are emitted.
149
+
150
+ We can say that together ``a``, ``b`` and ``c`` are the three shortest
151
+ geodesics. Furthermore, we can also say that the systole
152
+ of the manifold is in ``[1.0, 2.0]`` even though ``a`` itself might not be
153
+ the shortest geodesic. The latter is true in general:
154
+
155
+ **Verified systole**
156
+
157
+ It is not necessarily true that the first geodesic returned
158
+ by the method is the shortest geodesic. Despite this, the interval for
159
+ the real length of the first geodesic always contains the systole of
160
+ the manifold::
161
+
162
+ sage: M = Manifold("m004")
163
+ sage: spec = M.length_spectrum_alt_gen(verified=True)
164
+ sage: g = next(spec) # g might or might not be shortest geodesic
165
+ sage: systole = g.length.real() # But interval is large enough to contain systole
166
+ sage: systole # doctest: +NUMERIC6
167
+ 1.08707015?
168
+
169
+ :param bits_prec:
170
+ Precision used for the computation. Increase if computation did
171
+ not succeed.
172
+ :param verified:
173
+ Use :ref:`verified computation <verify-primer>`.
174
+ :return:
175
+ A generator to enumerate the geodesics such that the (lower bound
176
+ of the) real length is non-decreasing.
177
+ """
178
+
179
+ if not manifold.is_orientable():
180
+ raise NonorientableManifoldError(
181
+ "Manifold.length_spectrum_alt_gen", manifold)
182
+
183
+ # Triangulation with necessary geometric structures
184
+ mcomplex : Mcomplex = mcomplex_for_len_spec(
185
+ manifold,
186
+ bits_prec=bits_prec, verified=verified)
187
+ # Needed to know whether we convert a word such as [ 1 ] to
188
+ # a or to x1.
189
+ num_generators = len([g for g in mcomplex.GeneratorMatrices.keys()
190
+ if g > 0])
191
+ is_first = True
192
+
193
+ geodesic : GeodesicInfoBase
194
+ for geodesic in _length_spectrum_from_mcomplex(mcomplex, manifold):
195
+ # Convert information to user-friendly form
196
+ if isinstance(geodesic, CoreCurveGeodesicInfo):
197
+ core_curve = geodesic.core_curve
198
+ else:
199
+ core_curve = None
200
+ yield LengthSpectrumGeodesicInfo(
201
+ # _is_first indicates whether to print the header
202
+ _is_first = is_first,
203
+ length = geodesic.length,
204
+ # Convert word to a or x1
205
+ word = list_as_word(
206
+ simplify_geodesic_word(geodesic.word),
207
+ num_generators,
208
+ verbose_form=False),
209
+ core_curve = core_curve,
210
+ parity = 'orientation-preserving',
211
+ topology = 'circle',
212
+ multiplicity = 1)
213
+ is_first = False
214
+
215
+ def length_spectrum_alt(manifold,
216
+ count : Optional[int] = None,
217
+ max_len : Optional[Any] = None,
218
+ bits_prec : Optional[int] = None,
219
+ verified : bool = False
220
+ ) -> List[LengthSpectrumGeodesicInfo]:
221
+ """
222
+ Returns a list of geodesics. How far this list goes can be specified
223
+ by either a cut-off length or a count. The method only supports
224
+ orientable manifolds. It is a convenience method for
225
+ :meth:`~snappy.Manifold.length_spectrum_alt_gen`.
226
+ We refer the reader to
227
+ :meth:`~snappy.Manifold.length_spectrum_alt_gen`
228
+ for further details not covered here.
229
+
230
+ **Cut-off length**
231
+
232
+ Here is an example where a cut-off length for the geodesics is specified::
233
+
234
+ >>> M = Manifold("m202(3,4)(3,4)")
235
+ >>> M.length_spectrum_alt(max_len = 0.5) # doctest: +SKIP
236
+ [Length Core curve Word
237
+ 0.14820741547094 - 1.76955170166922*I Cusp 1 bcDc,
238
+ 0.14820741547097 - 1.76955170166923*I Cusp 0 aabcDabcB]
239
+
240
+ It also supports :ref:`verified <verify-primer>` computations::
241
+
242
+ sage: M.length_spectrum_alt(max_len = 0.5, verified = True, bits_prec = 100) # doctest: +SKIP
243
+ [Length Core curve Word
244
+ 0.148207415470948? - 1.76955170166924? *I Cusp 0 aabcDabcB,
245
+ 0.14820741547094... - 1.76955170166923...*I Cusp 1 bcDc]
246
+
247
+ If :attr:`verified=True`, the returned list is guaranteed to include all
248
+ geodesics up to the given cut-off length and might include additional
249
+ geodesics.
250
+
251
+ **Count**
252
+
253
+ Here is an example where a count is specified::
254
+
255
+ >>> M = Manifold("m202(3,4)(3,4)")
256
+ >>> M.length_spectrum_alt(count = 3) # doctest: +SKIP
257
+ [Length Core curve Word
258
+ 0.14820741547094 - 1.76955170166922*I Cusp 1 bcDc,
259
+ 0.14820741547097 - 1.76955170166923*I Cusp 0 aabcDabcB,
260
+ 0.79356651781096 + 2.65902431489655*I - aB,
261
+ 0.79356651781096 + 2.65902431489655*I - b]
262
+
263
+ Note that the number of geodesics listed might be larger than the given
264
+ count. In particular, this happens when the same (real) length appears
265
+ multiple times. If :attr:`verified=True`, the returned list is guaranteed
266
+ to include the :attr:`count` shortest geodesics and might include additional
267
+ geodesics.
268
+
269
+ **Verified systole**
270
+
271
+ Even though, the first reported geodesic might not be the shortest, we
272
+ obtain an interval containing the systole as follows, also see
273
+ :meth:`~snappy.Manifold.length_spectrum_alt_gen`::
274
+
275
+ sage: M = Manifold("m004")
276
+ sage: M.length_spectrum_alt(count=1, verified=True, bits_prec=100)[0].length.real() # doctest: +NUMERIC21
277
+ 1.0870701449957390997853?
278
+
279
+ :param count:
280
+ Number of shortest geodesics to list. The actual result might
281
+ contain additional geodesics. Exactly one of :attr:`count` and
282
+ :attr:`max_len` has to be specified.
283
+ :param max_len:
284
+ Cut-off length for geodesics. The actual result includes all
285
+ geodesics up to the given length and might include additional
286
+ geodesics. Exactly one of :attr:`count` and :attr:`max_len` has
287
+ to be specified.
288
+ :param bits_prec:
289
+ Precision used for the computation. Increase if computation did
290
+ not succeed.
291
+ :param verified:
292
+ Use :ref:`verified computation <verify-primer>`.
293
+ :return:
294
+ A list of geodesics such that the (lower bound of) the real
295
+ length is non-decreasing.
296
+ """
297
+
298
+ has_count = count is not None
299
+ has_max_len = max_len is not None
300
+
301
+ if not (has_count ^ has_max_len):
302
+ raise ValueError(
303
+ "Must specify exactly one of count or max_len.")
304
+
305
+ if has_max_len:
306
+ return list(
307
+ _length_spectrum_alt_max_len(
308
+ manifold,
309
+ max_len = max_len,
310
+ bits_prec = bits_prec,
311
+ verified = verified))
312
+ else:
313
+ return list(
314
+ _length_spectrum_alt_count(
315
+ manifold,
316
+ count = count,
317
+ bits_prec = bits_prec,
318
+ verified = verified))
319
+
320
+ def _length_spectrum_alt_max_len(
321
+ manifold, *,
322
+ max_len : Any,
323
+ bits_prec : Optional[int],
324
+ verified : bool):
325
+ """
326
+ Generator to produce all geodesics up to a given length.
327
+ """
328
+
329
+ if verified:
330
+ if not _within_sage:
331
+ raise SageNotAvailable('Sorry, this feature requires using SnapPy inside Sage.')
332
+ if bits_prec is None:
333
+ resolved_bits_prec = manifold._precision()
334
+ else:
335
+ resolved_bits_prec = bits_prec
336
+ RIF = RealIntervalField(resolved_bits_prec)
337
+ resolved_max_len = RIF(max_len)
338
+ else:
339
+ # A bit of fudge
340
+ resolved_max_len = max_len * 1.0000152
341
+
342
+ for info in length_spectrum_alt_gen(manifold = manifold,
343
+ bits_prec = bits_prec,
344
+ verified = verified):
345
+ # For verified computation:
346
+ #
347
+ # Recall that length_spectrum_alt_gen gives geodesics in the
348
+ # order such that the lower bound of info.length.real() is
349
+ # guaranteed to be (not strictly) increasing.
350
+ #
351
+ # The inequality checks that the lower bound of info.length.real()
352
+ # is larger than the given max_len.
353
+ #
354
+ # Thus, if true, any further geodesics will have length larger
355
+ # than max_len.
356
+ #
357
+ if info.length.real() > resolved_max_len:
358
+ break
359
+ yield info
360
+
361
+ def _length_spectrum_alt_count(
362
+ manifold, *,
363
+ count : int,
364
+ bits_prec : Optional[int],
365
+ verified : bool):
366
+ """
367
+ Generator to return a (potential) superset of geodesics containing
368
+ the count shortest geodesics.
369
+
370
+ In the verified case, this is a bit tricky since the only guarantee
371
+ we have is that the lower bound of the interval of the length
372
+ is (non-strictly) increasing.
373
+ """
374
+
375
+ if not count > 0:
376
+ # Sanitycheck.
377
+ return
378
+
379
+ for i, info in enumerate(
380
+ length_spectrum_alt_gen(manifold = manifold,
381
+ bits_prec = bits_prec,
382
+ verified = verified)):
383
+ this_len = info.length.real()
384
+
385
+ if i >= count:
386
+ if verified:
387
+ # The lower bound of the length of the current geodesic
388
+ # is larger than the maximum length of any geodesic
389
+ # encountered so far.
390
+ # Thus all following geodesics will be longer than any
391
+ # of geodesic encountered so far.
392
+ if this_len > max_len:
393
+ break
394
+ else:
395
+ # Add a fudge factor to account for geodesics with
396
+ # true equal length might have slightly different
397
+ # length due to numeric noise.
398
+ if this_len > 1.0000152 * max_len:
399
+ break
400
+
401
+ # Update the maximal length of geodesics encountered so far.
402
+ if i == 0:
403
+ max_len = this_len
404
+ else:
405
+ max_len = correct_max([max_len, this_len])
406
+
407
+ yield info
408
+
409
+ def _length_spectrum_from_mcomplex(
410
+ mcomplex : Mcomplex, manifold) -> Sequence[GeodesicInfoBase]:
411
+ """
412
+ Implements length_spectrum given an Mcomplex constructed with
413
+ mcomplex_for_len_spec and the SnapPy Manifold.
414
+ """
415
+
416
+ if mcomplex.verified:
417
+ epsilon = 0
418
+ else:
419
+ # A bit of extra tiling before outputting a geodesic.
420
+ epsilon = mcomplex.RF(0.0001)
421
+
422
+ # A set-like structure holding GeodesicKeyInfo's to record
423
+ # which geodesics were already emitted.
424
+ # Two geodesics are regarded as equal if one is a
425
+ # multiple of another (including inverse) and if they are
426
+ # conjugates.
427
+ #
428
+ # Note that the structure does not work correctly if we add
429
+ # a multiple of a geodesic to it first. If we add a geodesic
430
+ # that is not a multiple first though, trying to add a multiple of
431
+ # that geodesic will be detected and add will return False.
432
+ #
433
+ # The structure does not hold geodesics corresponding to
434
+ # core curves.
435
+ #
436
+ visited_geodesics = get_geodesic_key_info_set(mcomplex)
437
+
438
+ # A priority queue of geodesics (including those corresponding
439
+ # to core curves.). The key is the lower bound of the geodesic
440
+ # length.
441
+ #
442
+ # The reason for the queue is this:
443
+ # When tiling to find geodesics, the geodesics are not emitted in order.
444
+ # The tiling algorithm does tell us though that we have seen all geodesics
445
+ # up to a certain length.
446
+ #
447
+ # We will add geodesics to the priority queue as we tile.
448
+ # And we pick off a geodesic g from the top of the priority queue
449
+ # when we know that we have tiled far enough to see all geodesics that
450
+ # could be shorter than g.
451
+ pending_geodesics : Sequence[GeodesicInfoBase] = []
452
+
453
+ # Core curves are treated differently. We add them here to the priority
454
+ # queue explicitly and filter them out when tiling later.
455
+ for geodesic in _geodesic_key_infos_for_core_curves(
456
+ mcomplex, manifold):
457
+ heapq.heappush(pending_geodesics, geodesic)
458
+
459
+ # The real length of the last geodesic that we emitted.
460
+ last_real_length : Optional[Any] = None
461
+
462
+ for tile in compute_length_spectrum_tiles(mcomplex):
463
+
464
+ # We have tiled far enough to have seen all geodesics up to
465
+ # this length.
466
+ #
467
+ # Note that this comment about tiling only applies to the non-core
468
+ # curve geodesics. However, we have already added all core curves
469
+ # explicitly earlier, so we are fine.
470
+ #
471
+ r = tile.lower_bound_geodesic_length
472
+ while (pending_geodesics and
473
+ pending_geodesics[0].length.real() + epsilon < r):
474
+ # We can emit the geodesic g from the top of the priority queue
475
+ # - if we haven't emitted it already earlier.
476
+ # That is because all geodesics that we have not seen when
477
+ # tiling this far have length larger than g - or at least its
478
+ # lower bound is larger than than the lengths of those unseen
479
+ # geodesics.
480
+ geodesic : GeodesicInfoBase = heapq.heappop(pending_geodesics)
481
+ real_length = geodesic.length.real()
482
+
483
+ if _optimization:
484
+ # An optimization. If the geodesic has length less than
485
+ # what the last one, we know it is a duplicate.
486
+ if last_real_length is not None:
487
+ if last_real_length > real_length:
488
+ continue
489
+
490
+ if isinstance(geodesic, GeodesicKeyInfo):
491
+ # Handle the non-core curve case.
492
+ line : R13Line = geodesic.r13_line()
493
+
494
+ # If this geodesic does not intersect the spine, it is conjugate
495
+ # to a geodesic that we have already seen earlier, so we can
496
+ # skip it.
497
+ #
498
+ # Note that this also filters out parabolics. For them, the
499
+ # fixed points coincide and we get a degenerate line. We can
500
+ # still compute the distance to the spine center and it will
501
+ # be a very large number or interval with a large number as
502
+ # lower bound and infinity as upper bound.
503
+
504
+ # We first do a quick global check to see whether the spine
505
+ # intersects a ball containing the spine.
506
+ if distance_r13_point_line(
507
+ mcomplex.spine_center, line) > mcomplex.spine_radius:
508
+ continue
509
+
510
+ # And then a similar check per tetrahedron.
511
+ if all(distance_r13_point_line(
512
+ tet.spine_center, line) > tet.spine_radius
513
+ for tet in mcomplex.Tetrahedra):
514
+ continue
515
+
516
+ if mcomplex.verified:
517
+ # We need to make sure that the intervals are tight enough
518
+ # that we do not accidentally add a multiple before adding
519
+ # a primitive.
520
+ if not real_length.relative_diameter() < 0.125:
521
+ raise InsufficientPrecisionError(
522
+ "Interval of real length of geodesic too large. "
523
+ "Increase precision to fix it.")
524
+
525
+ # We also ignore core curves here since they have been
526
+ # explicitly added earlier.
527
+ #
528
+ # Note that this requires computing the start point info
529
+ # which can be expensive so we do the other checks first.
530
+ #
531
+ if geodesic.geodesic_start_point_info().core_curve_cusp:
532
+ continue
533
+
534
+ # Check that we have not visited this geodesic already
535
+ if not visited_geodesics.add(geodesic):
536
+ continue
537
+ # else:
538
+ # This is a core curve. We already know that they are
539
+ # de-duplicated (and cannot be parabolics), so no checks
540
+ # to perform.
541
+
542
+ if not real_length > 0:
543
+ raise InsufficientPrecisionError(
544
+ "Could not verify that a geodesic is not a parabolic. "
545
+ "This is most likely due to insufficient precision.")
546
+
547
+ if last_real_length is not None:
548
+ if not lower(last_real_length) <= lower(real_length):
549
+ raise InsufficientPrecisionError(
550
+ "Lower bound of new geodesics is less than that of "
551
+ "already seen geodesics. Re-start length spectrum "
552
+ "computation with higher precision to see more "
553
+ "geodesics.")
554
+
555
+ last_real_length = real_length
556
+
557
+ yield geodesic
558
+
559
+ # Geodesic is processed, remove from priority queue.
560
+ heapq.heappush(
561
+ pending_geodesics,
562
+ GeodesicKeyInfo(mcomplex, tile.word, tile.o13_matrix))
563
+
564
+ def _geodesic_key_infos_for_core_curves(
565
+ mcomplex, manifold) -> Sequence[CoreCurveGeodesicInfo]:
566
+ """
567
+ Computes geodesic info for all core curves given Mcomplex constructed
568
+ with mcomplex_for_len_spec and the SnapPy Manifold.
569
+ """
570
+
571
+ G = manifold.fundamental_group(False)
572
+ all_peripheral_words = G.peripheral_curves(as_int_list = True)
573
+
574
+ for cusp, peripheral_words in zip(mcomplex.Vertices, all_peripheral_words):
575
+ if cusp.is_complete:
576
+ continue
577
+
578
+ word = []
579
+
580
+ # Add suitable multiples of word corresponding to meridian and longitude
581
+ for peripheral_word, f in zip(peripheral_words, cusp.filling_matrix[1]):
582
+ if f == 0:
583
+ continue
584
+ if f < 0:
585
+ peripheral_word = inverse_list_word(peripheral_word)
586
+ f = -f
587
+ word += f * peripheral_word
588
+
589
+ psl2c_matrix = word_list_to_psl2c_matrix(mcomplex, word)
590
+
591
+ o13_matrix = psl2c_to_o13(psl2c_matrix)
592
+
593
+ yield CoreCurveGeodesicInfo(
594
+ word,
595
+ o13_matrix,
596
+ core_curve = cusp.Index)