superacli 1.1.4 → 1.1.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (961) hide show
  1. package/__tests__/adapter-schema.test.js +2 -0
  2. package/__tests__/config.test.js +62 -1
  3. package/__tests__/help-json.test.js +2 -0
  4. package/__tests__/mcp-adapter.test.js +14 -4
  5. package/__tests__/mcp-local.test.js +159 -0
  6. package/__tests__/mcp-stdio-jsonrpc.test.js +105 -0
  7. package/__tests__/monty-plugin.test.js +121 -0
  8. package/__tests__/plugin-browser-use-uninstall.test.js +23 -0
  9. package/__tests__/plugin-browser-use.test.js +77 -0
  10. package/__tests__/plugins-command.test.js +92 -1
  11. package/__tests__/plugins-learn.test.js +62 -0
  12. package/__tests__/plugins-registry.test.js +3 -1
  13. package/__tests__/resend-plugin.test.js +122 -0
  14. package/__tests__/skills.test.js +4 -0
  15. package/cli/adapter-schema.js +3 -2
  16. package/cli/adapters/mcp.js +22 -3
  17. package/cli/adapters/process.js +34 -7
  18. package/cli/config.js +27 -1
  19. package/cli/help-json.js +2 -2
  20. package/cli/mcp-diagnostics.js +152 -0
  21. package/cli/mcp-discovery.js +221 -0
  22. package/cli/mcp-local.js +267 -25
  23. package/cli/mcp-stdio-jsonrpc.js +246 -0
  24. package/cli/plugin-install-guidance.js +25 -0
  25. package/cli/plugins-command.js +86 -3
  26. package/cli/plugins-learn.js +177 -0
  27. package/cli/plugins-manager.js +3 -0
  28. package/cli/plugins-registry.js +2 -1
  29. package/cli/skills-mcp.js +102 -0
  30. package/cli/skills.js +6 -40
  31. package/cli/supercli.js +7 -2
  32. package/docs/initial/mcp-local-mode.md +35 -0
  33. package/docs/mcp-cheatsheet.md +324 -0
  34. package/docs/plugins.md +7 -0
  35. package/package.json +1 -1
  36. package/plugins/browser-use/plugin.json +23 -0
  37. package/plugins/browser-use/scripts/post-install.js +146 -0
  38. package/plugins/browser-use/scripts/post-uninstall.js +28 -0
  39. package/plugins/browser-use/skills/quickstart/SKILL.md +47 -0
  40. package/plugins/monty/README.md +49 -0
  41. package/plugins/monty/plugin.json +69 -0
  42. package/plugins/monty/scripts/post-install.js +73 -0
  43. package/plugins/monty/scripts/post-uninstall.js +23 -0
  44. package/plugins/monty/scripts/run-python.js +140 -0
  45. package/plugins/monty/scripts/setup-monty.js +27 -0
  46. package/plugins/plugins.json +29 -0
  47. package/plugins/resend/plugin.json +371 -0
  48. package/plugins/resend/scripts/post-install.js +59 -0
  49. package/plugins/resend/scripts/post-uninstall.js +23 -0
  50. package/plugins/resend/scripts/setup-resend.js +27 -0
  51. package/plugins/resend/skills/quickstart/SKILL.md +80 -0
  52. package/ref-monty/.cargo/config.toml +3 -0
  53. package/ref-monty/.claude/settings.json +60 -0
  54. package/ref-monty/.claude/skills/fastmod/SKILL.md +22 -0
  55. package/ref-monty/.claude/skills/python-playground/SKILL.md +47 -0
  56. package/ref-monty/.codecov.yml +12 -0
  57. package/ref-monty/.github/actions/build-pgo-wheel/action.yml +72 -0
  58. package/ref-monty/.github/workflows/ci.yml +776 -0
  59. package/ref-monty/.github/workflows/codspeed.yml +45 -0
  60. package/ref-monty/.github/workflows/init-npm-packages.yml +82 -0
  61. package/ref-monty/.pre-commit-config.yaml +47 -0
  62. package/ref-monty/.python-version +1 -0
  63. package/ref-monty/.rustfmt.toml +4 -0
  64. package/ref-monty/.zed/settings.json +11 -0
  65. package/ref-monty/CLAUDE.md +535 -0
  66. package/ref-monty/Cargo.lock +3798 -0
  67. package/ref-monty/Cargo.toml +87 -0
  68. package/ref-monty/LICENSE +21 -0
  69. package/ref-monty/Makefile +216 -0
  70. package/ref-monty/README.md +430 -0
  71. package/ref-monty/RELEASING.md +47 -0
  72. package/ref-monty/crates/fuzz/Cargo.toml +30 -0
  73. package/ref-monty/crates/fuzz/fuzz_targets/string_input_panic.rs +37 -0
  74. package/ref-monty/crates/fuzz/fuzz_targets/tokens_input_panic.rs +552 -0
  75. package/ref-monty/crates/monty/Cargo.toml +68 -0
  76. package/ref-monty/crates/monty/benches/main.rs +247 -0
  77. package/ref-monty/crates/monty/build.rs +10 -0
  78. package/ref-monty/crates/monty/src/args.rs +733 -0
  79. package/ref-monty/crates/monty/src/asyncio.rs +179 -0
  80. package/ref-monty/crates/monty/src/builtins/abs.rs +55 -0
  81. package/ref-monty/crates/monty/src/builtins/all.rs +30 -0
  82. package/ref-monty/crates/monty/src/builtins/any.rs +30 -0
  83. package/ref-monty/crates/monty/src/builtins/bin.rs +59 -0
  84. package/ref-monty/crates/monty/src/builtins/chr.rs +46 -0
  85. package/ref-monty/crates/monty/src/builtins/divmod.rs +164 -0
  86. package/ref-monty/crates/monty/src/builtins/enumerate.rs +52 -0
  87. package/ref-monty/crates/monty/src/builtins/filter.rs +67 -0
  88. package/ref-monty/crates/monty/src/builtins/getattr.rs +65 -0
  89. package/ref-monty/crates/monty/src/builtins/hash.rs +28 -0
  90. package/ref-monty/crates/monty/src/builtins/hex.rs +58 -0
  91. package/ref-monty/crates/monty/src/builtins/id.rs +24 -0
  92. package/ref-monty/crates/monty/src/builtins/isinstance.rs +68 -0
  93. package/ref-monty/crates/monty/src/builtins/len.rs +25 -0
  94. package/ref-monty/crates/monty/src/builtins/map.rs +98 -0
  95. package/ref-monty/crates/monty/src/builtins/min_max.rs +113 -0
  96. package/ref-monty/crates/monty/src/builtins/mod.rs +246 -0
  97. package/ref-monty/crates/monty/src/builtins/next.rs +21 -0
  98. package/ref-monty/crates/monty/src/builtins/oct.rs +59 -0
  99. package/ref-monty/crates/monty/src/builtins/ord.rs +67 -0
  100. package/ref-monty/crates/monty/src/builtins/pow.rs +365 -0
  101. package/ref-monty/crates/monty/src/builtins/print.rs +141 -0
  102. package/ref-monty/crates/monty/src/builtins/repr.rs +16 -0
  103. package/ref-monty/crates/monty/src/builtins/reversed.rs +28 -0
  104. package/ref-monty/crates/monty/src/builtins/round.rs +174 -0
  105. package/ref-monty/crates/monty/src/builtins/sorted.rs +151 -0
  106. package/ref-monty/crates/monty/src/builtins/sum.rs +66 -0
  107. package/ref-monty/crates/monty/src/builtins/type_.rs +16 -0
  108. package/ref-monty/crates/monty/src/builtins/zip.rs +77 -0
  109. package/ref-monty/crates/monty/src/bytecode/builder.rs +699 -0
  110. package/ref-monty/crates/monty/src/bytecode/code.rs +310 -0
  111. package/ref-monty/crates/monty/src/bytecode/compiler.rs +3206 -0
  112. package/ref-monty/crates/monty/src/bytecode/mod.rs +24 -0
  113. package/ref-monty/crates/monty/src/bytecode/op.rs +617 -0
  114. package/ref-monty/crates/monty/src/bytecode/vm/async_exec.rs +1058 -0
  115. package/ref-monty/crates/monty/src/bytecode/vm/attr.rs +63 -0
  116. package/ref-monty/crates/monty/src/bytecode/vm/binary.rs +487 -0
  117. package/ref-monty/crates/monty/src/bytecode/vm/call.rs +767 -0
  118. package/ref-monty/crates/monty/src/bytecode/vm/collections.rs +741 -0
  119. package/ref-monty/crates/monty/src/bytecode/vm/compare.rs +147 -0
  120. package/ref-monty/crates/monty/src/bytecode/vm/exceptions.rs +297 -0
  121. package/ref-monty/crates/monty/src/bytecode/vm/format.rs +132 -0
  122. package/ref-monty/crates/monty/src/bytecode/vm/mod.rs +1958 -0
  123. package/ref-monty/crates/monty/src/bytecode/vm/scheduler.rs +620 -0
  124. package/ref-monty/crates/monty/src/exception_private.rs +1513 -0
  125. package/ref-monty/crates/monty/src/exception_public.rs +346 -0
  126. package/ref-monty/crates/monty/src/expressions.rs +694 -0
  127. package/ref-monty/crates/monty/src/fstring.rs +854 -0
  128. package/ref-monty/crates/monty/src/function.rs +119 -0
  129. package/ref-monty/crates/monty/src/heap.rs +1073 -0
  130. package/ref-monty/crates/monty/src/heap_data.rs +985 -0
  131. package/ref-monty/crates/monty/src/heap_traits.rs +312 -0
  132. package/ref-monty/crates/monty/src/intern.rs +837 -0
  133. package/ref-monty/crates/monty/src/io.rs +106 -0
  134. package/ref-monty/crates/monty/src/lib.rs +52 -0
  135. package/ref-monty/crates/monty/src/modules/asyncio.rs +144 -0
  136. package/ref-monty/crates/monty/src/modules/math.rs +1453 -0
  137. package/ref-monty/crates/monty/src/modules/mod.rs +120 -0
  138. package/ref-monty/crates/monty/src/modules/os.rs +116 -0
  139. package/ref-monty/crates/monty/src/modules/pathlib.rs +33 -0
  140. package/ref-monty/crates/monty/src/modules/re.rs +606 -0
  141. package/ref-monty/crates/monty/src/modules/sys.rs +60 -0
  142. package/ref-monty/crates/monty/src/modules/typing.rs +70 -0
  143. package/ref-monty/crates/monty/src/namespace.rs +21 -0
  144. package/ref-monty/crates/monty/src/object.rs +1040 -0
  145. package/ref-monty/crates/monty/src/os.rs +215 -0
  146. package/ref-monty/crates/monty/src/parse.rs +1730 -0
  147. package/ref-monty/crates/monty/src/prepare.rs +3015 -0
  148. package/ref-monty/crates/monty/src/repl.rs +1109 -0
  149. package/ref-monty/crates/monty/src/resource.rs +559 -0
  150. package/ref-monty/crates/monty/src/run.rs +457 -0
  151. package/ref-monty/crates/monty/src/run_progress.rs +821 -0
  152. package/ref-monty/crates/monty/src/signature.rs +651 -0
  153. package/ref-monty/crates/monty/src/sorting.rs +100 -0
  154. package/ref-monty/crates/monty/src/types/bytes.rs +2356 -0
  155. package/ref-monty/crates/monty/src/types/dataclass.rs +345 -0
  156. package/ref-monty/crates/monty/src/types/dict.rs +879 -0
  157. package/ref-monty/crates/monty/src/types/dict_view.rs +619 -0
  158. package/ref-monty/crates/monty/src/types/iter.rs +799 -0
  159. package/ref-monty/crates/monty/src/types/list.rs +929 -0
  160. package/ref-monty/crates/monty/src/types/long_int.rs +211 -0
  161. package/ref-monty/crates/monty/src/types/mod.rs +48 -0
  162. package/ref-monty/crates/monty/src/types/module.rs +146 -0
  163. package/ref-monty/crates/monty/src/types/namedtuple.rs +261 -0
  164. package/ref-monty/crates/monty/src/types/path.rs +596 -0
  165. package/ref-monty/crates/monty/src/types/property.rs +35 -0
  166. package/ref-monty/crates/monty/src/types/py_trait.rs +322 -0
  167. package/ref-monty/crates/monty/src/types/range.rs +285 -0
  168. package/ref-monty/crates/monty/src/types/re_match.rs +522 -0
  169. package/ref-monty/crates/monty/src/types/re_pattern.rs +726 -0
  170. package/ref-monty/crates/monty/src/types/set.rs +1373 -0
  171. package/ref-monty/crates/monty/src/types/slice.rs +257 -0
  172. package/ref-monty/crates/monty/src/types/str.rs +2051 -0
  173. package/ref-monty/crates/monty/src/types/tuple.rs +376 -0
  174. package/ref-monty/crates/monty/src/types/type.rs +407 -0
  175. package/ref-monty/crates/monty/src/value.rs +2558 -0
  176. package/ref-monty/crates/monty/test_cases/args__dict_get_no_args.py +3 -0
  177. package/ref-monty/crates/monty/test_cases/args__dict_get_too_many.py +3 -0
  178. package/ref-monty/crates/monty/test_cases/args__dict_items_with_args.py +3 -0
  179. package/ref-monty/crates/monty/test_cases/args__dict_keys_with_args.py +3 -0
  180. package/ref-monty/crates/monty/test_cases/args__dict_pop_no_args.py +3 -0
  181. package/ref-monty/crates/monty/test_cases/args__dict_pop_too_many.py +3 -0
  182. package/ref-monty/crates/monty/test_cases/args__dict_values_with_args.py +3 -0
  183. package/ref-monty/crates/monty/test_cases/args__id_too_many.py +2 -0
  184. package/ref-monty/crates/monty/test_cases/args__len_no_args.py +2 -0
  185. package/ref-monty/crates/monty/test_cases/args__len_too_many.py +2 -0
  186. package/ref-monty/crates/monty/test_cases/args__len_type_error_int.py +9 -0
  187. package/ref-monty/crates/monty/test_cases/args__len_type_error_none.py +9 -0
  188. package/ref-monty/crates/monty/test_cases/args__list_append_no_args.py +3 -0
  189. package/ref-monty/crates/monty/test_cases/args__list_append_too_many.py +3 -0
  190. package/ref-monty/crates/monty/test_cases/args__list_insert_too_few.py +3 -0
  191. package/ref-monty/crates/monty/test_cases/args__list_insert_too_many.py +3 -0
  192. package/ref-monty/crates/monty/test_cases/args__repr_no_args.py +2 -0
  193. package/ref-monty/crates/monty/test_cases/arith__div_zero_float.py +2 -0
  194. package/ref-monty/crates/monty/test_cases/arith__div_zero_int.py +2 -0
  195. package/ref-monty/crates/monty/test_cases/arith__floordiv_zero_float.py +2 -0
  196. package/ref-monty/crates/monty/test_cases/arith__floordiv_zero_int.py +2 -0
  197. package/ref-monty/crates/monty/test_cases/arith__pow_zero_neg.py +2 -0
  198. package/ref-monty/crates/monty/test_cases/arith__pow_zero_neg_builtin.py +9 -0
  199. package/ref-monty/crates/monty/test_cases/assert__expr_fail.py +2 -0
  200. package/ref-monty/crates/monty/test_cases/assert__fail.py +2 -0
  201. package/ref-monty/crates/monty/test_cases/assert__fail_msg.py +2 -0
  202. package/ref-monty/crates/monty/test_cases/assert__fn_fail.py +3 -0
  203. package/ref-monty/crates/monty/test_cases/assert__ops.py +11 -0
  204. package/ref-monty/crates/monty/test_cases/async__asyncio_run.py +47 -0
  205. package/ref-monty/crates/monty/test_cases/async__basic.py +10 -0
  206. package/ref-monty/crates/monty/test_cases/async__closure.py +14 -0
  207. package/ref-monty/crates/monty/test_cases/async__double_await_coroutine.py +16 -0
  208. package/ref-monty/crates/monty/test_cases/async__exception.py +10 -0
  209. package/ref-monty/crates/monty/test_cases/async__ext_call.py +73 -0
  210. package/ref-monty/crates/monty/test_cases/async__gather_all.py +85 -0
  211. package/ref-monty/crates/monty/test_cases/async__nested_await.py +15 -0
  212. package/ref-monty/crates/monty/test_cases/async__nested_gather_ext.py +37 -0
  213. package/ref-monty/crates/monty/test_cases/async__not_awaitable.py +10 -0
  214. package/ref-monty/crates/monty/test_cases/async__not_imported.py +14 -0
  215. package/ref-monty/crates/monty/test_cases/async__recursion_depth_isolation.py +27 -0
  216. package/ref-monty/crates/monty/test_cases/async__return_types.py +31 -0
  217. package/ref-monty/crates/monty/test_cases/async__sequential.py +16 -0
  218. package/ref-monty/crates/monty/test_cases/async__traceback.py +19 -0
  219. package/ref-monty/crates/monty/test_cases/async__with_args.py +14 -0
  220. package/ref-monty/crates/monty/test_cases/attr__get_int_error.py +9 -0
  221. package/ref-monty/crates/monty/test_cases/attr__get_list_error.py +9 -0
  222. package/ref-monty/crates/monty/test_cases/attr__set_frozen_nonfield.py +12 -0
  223. package/ref-monty/crates/monty/test_cases/attr__set_int_error.py +10 -0
  224. package/ref-monty/crates/monty/test_cases/attr__set_list_error.py +10 -0
  225. package/ref-monty/crates/monty/test_cases/bench__kitchen_sink.py +68 -0
  226. package/ref-monty/crates/monty/test_cases/bool__ops.py +20 -0
  227. package/ref-monty/crates/monty/test_cases/builtin__add_type_error.py +2 -0
  228. package/ref-monty/crates/monty/test_cases/builtin__filter.py +62 -0
  229. package/ref-monty/crates/monty/test_cases/builtin__filter_not_iterable.py +11 -0
  230. package/ref-monty/crates/monty/test_cases/builtin__getattr.py +84 -0
  231. package/ref-monty/crates/monty/test_cases/builtin__iter_funcs.py +42 -0
  232. package/ref-monty/crates/monty/test_cases/builtin__iter_next.py +66 -0
  233. package/ref-monty/crates/monty/test_cases/builtin__map.py +74 -0
  234. package/ref-monty/crates/monty/test_cases/builtin__map_not_iterable.py +11 -0
  235. package/ref-monty/crates/monty/test_cases/builtin__math_funcs.py +154 -0
  236. package/ref-monty/crates/monty/test_cases/builtin__more_iter_funcs.py +148 -0
  237. package/ref-monty/crates/monty/test_cases/builtin__next_stop_iteration.py +10 -0
  238. package/ref-monty/crates/monty/test_cases/builtin__print_invalid_kwarg.py +9 -0
  239. package/ref-monty/crates/monty/test_cases/builtin__print_kwargs.py +12 -0
  240. package/ref-monty/crates/monty/test_cases/builtin__repr.py +3 -0
  241. package/ref-monty/crates/monty/test_cases/builtin__string_funcs.py +73 -0
  242. package/ref-monty/crates/monty/test_cases/bytes__decode_invalid_utf8.py +18 -0
  243. package/ref-monty/crates/monty/test_cases/bytes__endswith_str_error.py +10 -0
  244. package/ref-monty/crates/monty/test_cases/bytes__getitem_index_error.py +10 -0
  245. package/ref-monty/crates/monty/test_cases/bytes__index_start_gt_end.py +10 -0
  246. package/ref-monty/crates/monty/test_cases/bytes__methods.py +394 -0
  247. package/ref-monty/crates/monty/test_cases/bytes__negative_count.py +9 -0
  248. package/ref-monty/crates/monty/test_cases/bytes__ops.py +90 -0
  249. package/ref-monty/crates/monty/test_cases/bytes__startswith_str_error.py +10 -0
  250. package/ref-monty/crates/monty/test_cases/call_object.py +3 -0
  251. package/ref-monty/crates/monty/test_cases/chain_comparison__all.py +79 -0
  252. package/ref-monty/crates/monty/test_cases/closure__param_shadows_outer.py +81 -0
  253. package/ref-monty/crates/monty/test_cases/closure__pep448.py +203 -0
  254. package/ref-monty/crates/monty/test_cases/closure__undefined_nonlocal.py +13 -0
  255. package/ref-monty/crates/monty/test_cases/compare__mixed_types.py +120 -0
  256. package/ref-monty/crates/monty/test_cases/comprehension__all.py +208 -0
  257. package/ref-monty/crates/monty/test_cases/comprehension__scope.py +7 -0
  258. package/ref-monty/crates/monty/test_cases/comprehension__unbound_local.py +14 -0
  259. package/ref-monty/crates/monty/test_cases/dataclass__basic.py +238 -0
  260. package/ref-monty/crates/monty/test_cases/dataclass__call_field_error.py +12 -0
  261. package/ref-monty/crates/monty/test_cases/dataclass__frozen_set_error.py +12 -0
  262. package/ref-monty/crates/monty/test_cases/dataclass__get_missing_attr_error.py +11 -0
  263. package/ref-monty/crates/monty/test_cases/dict__get_unhashable_key.py +3 -0
  264. package/ref-monty/crates/monty/test_cases/dict__literal_unhashable_key.py +2 -0
  265. package/ref-monty/crates/monty/test_cases/dict__method_pop_missing_error.py +3 -0
  266. package/ref-monty/crates/monty/test_cases/dict__methods.py +151 -0
  267. package/ref-monty/crates/monty/test_cases/dict__ops.py +133 -0
  268. package/ref-monty/crates/monty/test_cases/dict__pop_unhashable_key.py +4 -0
  269. package/ref-monty/crates/monty/test_cases/dict__popitem_empty.py +9 -0
  270. package/ref-monty/crates/monty/test_cases/dict__subscript_missing_key.py +3 -0
  271. package/ref-monty/crates/monty/test_cases/dict__unhashable_dict_key.py +2 -0
  272. package/ref-monty/crates/monty/test_cases/dict__unhashable_list_key.py +2 -0
  273. package/ref-monty/crates/monty/test_cases/dict__unpack_type_error.py +2 -0
  274. package/ref-monty/crates/monty/test_cases/dict__views.py +165 -0
  275. package/ref-monty/crates/monty/test_cases/edge__all.py +26 -0
  276. package/ref-monty/crates/monty/test_cases/edge__float_int_mod.py +2 -0
  277. package/ref-monty/crates/monty/test_cases/edge__int_float_mod.py +2 -0
  278. package/ref-monty/crates/monty/test_cases/exc__args.py +16 -0
  279. package/ref-monty/crates/monty/test_cases/exc__str.py +15 -0
  280. package/ref-monty/crates/monty/test_cases/execute_ok__all.py +54 -0
  281. package/ref-monty/crates/monty/test_cases/execute_raise__error_instance_str.py +2 -0
  282. package/ref-monty/crates/monty/test_cases/execute_raise__error_no_args.py +2 -0
  283. package/ref-monty/crates/monty/test_cases/execute_raise__error_string_arg.py +2 -0
  284. package/ref-monty/crates/monty/test_cases/execute_raise__error_string_arg_quotes.py +2 -0
  285. package/ref-monty/crates/monty/test_cases/execute_raise__error_type.py +2 -0
  286. package/ref-monty/crates/monty/test_cases/execute_raise__raise_instance_via_var.py +4 -0
  287. package/ref-monty/crates/monty/test_cases/execute_raise__raise_list.py +2 -0
  288. package/ref-monty/crates/monty/test_cases/execute_raise__raise_number.py +2 -0
  289. package/ref-monty/crates/monty/test_cases/execute_raise__raise_type_call_via_var.py +4 -0
  290. package/ref-monty/crates/monty/test_cases/execute_raise__raise_type_direct.py +3 -0
  291. package/ref-monty/crates/monty/test_cases/execute_raise__raise_type_via_var.py +4 -0
  292. package/ref-monty/crates/monty/test_cases/ext_call__arg_side_effect_bug.py +22 -0
  293. package/ref-monty/crates/monty/test_cases/ext_call__augmented.py +17 -0
  294. package/ref-monty/crates/monty/test_cases/ext_call__augmented_refcount_bug.py +7 -0
  295. package/ref-monty/crates/monty/test_cases/ext_call__bare_raise_after_resume.py +34 -0
  296. package/ref-monty/crates/monty/test_cases/ext_call__basic.py +99 -0
  297. package/ref-monty/crates/monty/test_cases/ext_call__boolean.py +37 -0
  298. package/ref-monty/crates/monty/test_cases/ext_call__boolean_side_effect_hang.py +17 -0
  299. package/ref-monty/crates/monty/test_cases/ext_call__closure_bug.py +16 -0
  300. package/ref-monty/crates/monty/test_cases/ext_call__comparison.py +26 -0
  301. package/ref-monty/crates/monty/test_cases/ext_call__deep_call_stack.py +18 -0
  302. package/ref-monty/crates/monty/test_cases/ext_call__elif.py +171 -0
  303. package/ref-monty/crates/monty/test_cases/ext_call__exc.py +4 -0
  304. package/ref-monty/crates/monty/test_cases/ext_call__exc_deep_stack.py +39 -0
  305. package/ref-monty/crates/monty/test_cases/ext_call__exc_in_function.py +17 -0
  306. package/ref-monty/crates/monty/test_cases/ext_call__exc_nested_functions.py +31 -0
  307. package/ref-monty/crates/monty/test_cases/ext_call__ext_exc.py +171 -0
  308. package/ref-monty/crates/monty/test_cases/ext_call__for.py +114 -0
  309. package/ref-monty/crates/monty/test_cases/ext_call__fstring.py +12 -0
  310. package/ref-monty/crates/monty/test_cases/ext_call__if.py +135 -0
  311. package/ref-monty/crates/monty/test_cases/ext_call__if_condition.py +37 -0
  312. package/ref-monty/crates/monty/test_cases/ext_call__in_closure.py +14 -0
  313. package/ref-monty/crates/monty/test_cases/ext_call__in_function.py +40 -0
  314. package/ref-monty/crates/monty/test_cases/ext_call__in_function_simple.py +7 -0
  315. package/ref-monty/crates/monty/test_cases/ext_call__literals.py +17 -0
  316. package/ref-monty/crates/monty/test_cases/ext_call__multi_in_func.py +32 -0
  317. package/ref-monty/crates/monty/test_cases/ext_call__name_lookup.py +69 -0
  318. package/ref-monty/crates/monty/test_cases/ext_call__name_lookup_undefined.py +4 -0
  319. package/ref-monty/crates/monty/test_cases/ext_call__nested_calls.py +14 -0
  320. package/ref-monty/crates/monty/test_cases/ext_call__recursion_bug.py +19 -0
  321. package/ref-monty/crates/monty/test_cases/ext_call__return.py +28 -0
  322. package/ref-monty/crates/monty/test_cases/ext_call__side_effects.py +25 -0
  323. package/ref-monty/crates/monty/test_cases/ext_call__subscript.py +7 -0
  324. package/ref-monty/crates/monty/test_cases/ext_call__ternary.py +28 -0
  325. package/ref-monty/crates/monty/test_cases/ext_call__try.py +280 -0
  326. package/ref-monty/crates/monty/test_cases/ext_call__try_simple.py +10 -0
  327. package/ref-monty/crates/monty/test_cases/ext_call__unary.py +13 -0
  328. package/ref-monty/crates/monty/test_cases/frozenset__ops.py +178 -0
  329. package/ref-monty/crates/monty/test_cases/fstring__all.py +236 -0
  330. package/ref-monty/crates/monty/test_cases/fstring__error_eq_align_on_str.py +3 -0
  331. package/ref-monty/crates/monty/test_cases/fstring__error_float_f_on_str.py +3 -0
  332. package/ref-monty/crates/monty/test_cases/fstring__error_int_d_on_float.py +3 -0
  333. package/ref-monty/crates/monty/test_cases/fstring__error_int_d_on_str.py +3 -0
  334. package/ref-monty/crates/monty/test_cases/fstring__error_invalid_spec.py +4 -0
  335. package/ref-monty/crates/monty/test_cases/fstring__error_invalid_spec_dynamic.py +4 -0
  336. package/ref-monty/crates/monty/test_cases/fstring__error_invalid_spec_str.py +4 -0
  337. package/ref-monty/crates/monty/test_cases/fstring__error_str_s_on_int.py +3 -0
  338. package/ref-monty/crates/monty/test_cases/function__call_duplicate_kwargs.py +6 -0
  339. package/ref-monty/crates/monty/test_cases/function__call_unpack.py +42 -0
  340. package/ref-monty/crates/monty/test_cases/function__defaults.py +117 -0
  341. package/ref-monty/crates/monty/test_cases/function__err_duplicate_arg.py +7 -0
  342. package/ref-monty/crates/monty/test_cases/function__err_duplicate_first_arg.py +7 -0
  343. package/ref-monty/crates/monty/test_cases/function__err_duplicate_kwarg_cleanup.py +9 -0
  344. package/ref-monty/crates/monty/test_cases/function__err_kwonly_as_positional.py +7 -0
  345. package/ref-monty/crates/monty/test_cases/function__err_missing_all_posonly.py +7 -0
  346. package/ref-monty/crates/monty/test_cases/function__err_missing_heap_cleanup.py +9 -0
  347. package/ref-monty/crates/monty/test_cases/function__err_missing_kwonly.py +7 -0
  348. package/ref-monty/crates/monty/test_cases/function__err_missing_posonly_with_kwarg.py +7 -0
  349. package/ref-monty/crates/monty/test_cases/function__err_missing_with_posonly.py +7 -0
  350. package/ref-monty/crates/monty/test_cases/function__err_posonly_as_kwarg.py +7 -0
  351. package/ref-monty/crates/monty/test_cases/function__err_posonly_first_as_kwarg.py +7 -0
  352. package/ref-monty/crates/monty/test_cases/function__err_too_many_posonly.py +7 -0
  353. package/ref-monty/crates/monty/test_cases/function__err_too_many_with_kwonly.py +7 -0
  354. package/ref-monty/crates/monty/test_cases/function__err_unexpected_kwarg.py +7 -0
  355. package/ref-monty/crates/monty/test_cases/function__err_unexpected_kwarg_cleanup.py +9 -0
  356. package/ref-monty/crates/monty/test_cases/function__err_unexpected_kwarg_quote.py +13 -0
  357. package/ref-monty/crates/monty/test_cases/function__err_unexpected_kwarg_simple.py +7 -0
  358. package/ref-monty/crates/monty/test_cases/function__err_unpack_duplicate_arg.py +6 -0
  359. package/ref-monty/crates/monty/test_cases/function__err_unpack_duplicate_heap.py +8 -0
  360. package/ref-monty/crates/monty/test_cases/function__err_unpack_int.py +6 -0
  361. package/ref-monty/crates/monty/test_cases/function__err_unpack_nonstring_key.py +6 -0
  362. package/ref-monty/crates/monty/test_cases/function__err_unpack_not_mapping.py +6 -0
  363. package/ref-monty/crates/monty/test_cases/function__kwargs_unpacking.py +173 -0
  364. package/ref-monty/crates/monty/test_cases/function__ops.py +294 -0
  365. package/ref-monty/crates/monty/test_cases/function__return_none.py +42 -0
  366. package/ref-monty/crates/monty/test_cases/function__signatures.py +47 -0
  367. package/ref-monty/crates/monty/test_cases/function__too_few_args_all.py +6 -0
  368. package/ref-monty/crates/monty/test_cases/function__too_few_args_one.py +6 -0
  369. package/ref-monty/crates/monty/test_cases/function__too_few_args_two.py +6 -0
  370. package/ref-monty/crates/monty/test_cases/function__too_many_args_one.py +6 -0
  371. package/ref-monty/crates/monty/test_cases/function__too_many_args_two.py +6 -0
  372. package/ref-monty/crates/monty/test_cases/function__too_many_args_zero.py +6 -0
  373. package/ref-monty/crates/monty/test_cases/global__error_assigned_before.py +7 -0
  374. package/ref-monty/crates/monty/test_cases/global__ops.py +163 -0
  375. package/ref-monty/crates/monty/test_cases/hash__dict_unhashable.py +2 -0
  376. package/ref-monty/crates/monty/test_cases/hash__list_unhashable.py +2 -0
  377. package/ref-monty/crates/monty/test_cases/hash__ops.py +153 -0
  378. package/ref-monty/crates/monty/test_cases/id__bytes_literals_distinct.py +3 -0
  379. package/ref-monty/crates/monty/test_cases/id__int_copy_distinct.py +5 -0
  380. package/ref-monty/crates/monty/test_cases/id__is_number_is_number.py +3 -0
  381. package/ref-monty/crates/monty/test_cases/id__non_overlapping_lifetimes_distinct_types.py +10 -0
  382. package/ref-monty/crates/monty/test_cases/id__non_overlapping_lifetimes_same_types.py +6 -0
  383. package/ref-monty/crates/monty/test_cases/id__ops.py +97 -0
  384. package/ref-monty/crates/monty/test_cases/id__str_literals_same.py +3 -0
  385. package/ref-monty/crates/monty/test_cases/if__elif_else.py +207 -0
  386. package/ref-monty/crates/monty/test_cases/if__raise_elif.py +11 -0
  387. package/ref-monty/crates/monty/test_cases/if__raise_else.py +13 -0
  388. package/ref-monty/crates/monty/test_cases/if__raise_if.py +9 -0
  389. package/ref-monty/crates/monty/test_cases/if__raise_in_elif_condition.py +18 -0
  390. package/ref-monty/crates/monty/test_cases/if__raise_in_if_condition.py +16 -0
  391. package/ref-monty/crates/monty/test_cases/if_else_expr__all.py +55 -0
  392. package/ref-monty/crates/monty/test_cases/import__error_cannot_import.py +9 -0
  393. package/ref-monty/crates/monty/test_cases/import__error_module_not_found.py +9 -0
  394. package/ref-monty/crates/monty/test_cases/import__local_scope.py +68 -0
  395. package/ref-monty/crates/monty/test_cases/import__os.py +25 -0
  396. package/ref-monty/crates/monty/test_cases/import__relative_error.py +9 -0
  397. package/ref-monty/crates/monty/test_cases/import__relative_no_module_error.py +9 -0
  398. package/ref-monty/crates/monty/test_cases/import__runtime_error_when_executed.py +14 -0
  399. package/ref-monty/crates/monty/test_cases/import__star_error.py +11 -0
  400. package/ref-monty/crates/monty/test_cases/import__sys.py +47 -0
  401. package/ref-monty/crates/monty/test_cases/import__sys_monty.py +28 -0
  402. package/ref-monty/crates/monty/test_cases/import__type_checking_guard.py +37 -0
  403. package/ref-monty/crates/monty/test_cases/import__typing.py +25 -0
  404. package/ref-monty/crates/monty/test_cases/import__typing_type_ignore.py +4 -0
  405. package/ref-monty/crates/monty/test_cases/int__bigint.py +467 -0
  406. package/ref-monty/crates/monty/test_cases/int__bigint_errors.py +260 -0
  407. package/ref-monty/crates/monty/test_cases/int__ops.py +219 -0
  408. package/ref-monty/crates/monty/test_cases/int__overflow_division.py +84 -0
  409. package/ref-monty/crates/monty/test_cases/is_variant__all.py +36 -0
  410. package/ref-monty/crates/monty/test_cases/isinstance__arg2_list_error.py +2 -0
  411. package/ref-monty/crates/monty/test_cases/isinstance__arg2_type_error.py +2 -0
  412. package/ref-monty/crates/monty/test_cases/iter__dict_mutation.py +4 -0
  413. package/ref-monty/crates/monty/test_cases/iter__for.py +243 -0
  414. package/ref-monty/crates/monty/test_cases/iter__for_loop_unpacking.py +66 -0
  415. package/ref-monty/crates/monty/test_cases/iter__generator_expr.py +20 -0
  416. package/ref-monty/crates/monty/test_cases/iter__generator_expr_type.py +7 -0
  417. package/ref-monty/crates/monty/test_cases/iter__not_iterable.py +3 -0
  418. package/ref-monty/crates/monty/test_cases/lambda__all.py +145 -0
  419. package/ref-monty/crates/monty/test_cases/list__extend_not_iterable.py +7 -0
  420. package/ref-monty/crates/monty/test_cases/list__getitem_out_of_bounds.py +3 -0
  421. package/ref-monty/crates/monty/test_cases/list__index_not_found.py +9 -0
  422. package/ref-monty/crates/monty/test_cases/list__index_start_gt_end.py +10 -0
  423. package/ref-monty/crates/monty/test_cases/list__ops.py +473 -0
  424. package/ref-monty/crates/monty/test_cases/list__pop_empty.py +9 -0
  425. package/ref-monty/crates/monty/test_cases/list__pop_out_of_range.py +9 -0
  426. package/ref-monty/crates/monty/test_cases/list__pop_type_error.py +9 -0
  427. package/ref-monty/crates/monty/test_cases/list__remove_not_found.py +9 -0
  428. package/ref-monty/crates/monty/test_cases/list__setitem_dict_index.py +13 -0
  429. package/ref-monty/crates/monty/test_cases/list__setitem_huge_int_index.py +13 -0
  430. package/ref-monty/crates/monty/test_cases/list__setitem_index_error.py +10 -0
  431. package/ref-monty/crates/monty/test_cases/list__setitem_type_error.py +10 -0
  432. package/ref-monty/crates/monty/test_cases/list__unpack_type_error.py +2 -0
  433. package/ref-monty/crates/monty/test_cases/longint__index_error.py +3 -0
  434. package/ref-monty/crates/monty/test_cases/longint__repeat_error.py +3 -0
  435. package/ref-monty/crates/monty/test_cases/loop__break_continue.py +113 -0
  436. package/ref-monty/crates/monty/test_cases/loop__break_finally.py +69 -0
  437. package/ref-monty/crates/monty/test_cases/loop__break_in_function_error.py +13 -0
  438. package/ref-monty/crates/monty/test_cases/loop__break_in_if_error.py +11 -0
  439. package/ref-monty/crates/monty/test_cases/loop__break_nested_except_clears.py +55 -0
  440. package/ref-monty/crates/monty/test_cases/loop__break_outside_error.py +9 -0
  441. package/ref-monty/crates/monty/test_cases/loop__continue_finally.py +81 -0
  442. package/ref-monty/crates/monty/test_cases/loop__continue_in_function_error.py +13 -0
  443. package/ref-monty/crates/monty/test_cases/loop__continue_in_if_error.py +11 -0
  444. package/ref-monty/crates/monty/test_cases/loop__continue_nested_except_clears.py +60 -0
  445. package/ref-monty/crates/monty/test_cases/loop__continue_outside_error.py +9 -0
  446. package/ref-monty/crates/monty/test_cases/math__acos_domain_error.py +11 -0
  447. package/ref-monty/crates/monty/test_cases/math__acosh_domain_error.py +11 -0
  448. package/ref-monty/crates/monty/test_cases/math__asin_domain_error.py +11 -0
  449. package/ref-monty/crates/monty/test_cases/math__atanh_domain_error.py +11 -0
  450. package/ref-monty/crates/monty/test_cases/math__cos_inf_error.py +11 -0
  451. package/ref-monty/crates/monty/test_cases/math__cosh_overflow_error.py +11 -0
  452. package/ref-monty/crates/monty/test_cases/math__exp_overflow_error.py +11 -0
  453. package/ref-monty/crates/monty/test_cases/math__factorial_float_error.py +11 -0
  454. package/ref-monty/crates/monty/test_cases/math__factorial_negative_error.py +11 -0
  455. package/ref-monty/crates/monty/test_cases/math__floor_inf_error.py +11 -0
  456. package/ref-monty/crates/monty/test_cases/math__floor_nan_error.py +11 -0
  457. package/ref-monty/crates/monty/test_cases/math__floor_str_error.py +11 -0
  458. package/ref-monty/crates/monty/test_cases/math__fmod_inf_error.py +11 -0
  459. package/ref-monty/crates/monty/test_cases/math__gamma_neg_int_error.py +11 -0
  460. package/ref-monty/crates/monty/test_cases/math__gcd_float_error.py +11 -0
  461. package/ref-monty/crates/monty/test_cases/math__isqrt_negative_error.py +11 -0
  462. package/ref-monty/crates/monty/test_cases/math__ldexp_overflow_error.py +11 -0
  463. package/ref-monty/crates/monty/test_cases/math__log1p_domain_error.py +11 -0
  464. package/ref-monty/crates/monty/test_cases/math__log_base1_error.py +11 -0
  465. package/ref-monty/crates/monty/test_cases/math__log_zero_error.py +11 -0
  466. package/ref-monty/crates/monty/test_cases/math__module.py +1432 -0
  467. package/ref-monty/crates/monty/test_cases/math__pow_domain_error.py +11 -0
  468. package/ref-monty/crates/monty/test_cases/math__sin_inf_error.py +11 -0
  469. package/ref-monty/crates/monty/test_cases/math__sqrt_negative_error.py +11 -0
  470. package/ref-monty/crates/monty/test_cases/math__tan_inf_error.py +11 -0
  471. package/ref-monty/crates/monty/test_cases/math__trunc_str_error.py +11 -0
  472. package/ref-monty/crates/monty/test_cases/method__args_kwargs_unpacking.py +259 -0
  473. package/ref-monty/crates/monty/test_cases/name_error__unbound_local_func.py +19 -0
  474. package/ref-monty/crates/monty/test_cases/name_error__unbound_local_module.py +12 -0
  475. package/ref-monty/crates/monty/test_cases/name_error__undefined_call_chained.py +9 -0
  476. package/ref-monty/crates/monty/test_cases/name_error__undefined_call_in_expr.py +9 -0
  477. package/ref-monty/crates/monty/test_cases/name_error__undefined_call_in_function.py +16 -0
  478. package/ref-monty/crates/monty/test_cases/name_error__undefined_call_with_args.py +9 -0
  479. package/ref-monty/crates/monty/test_cases/name_error__undefined_global.py +10 -0
  480. package/ref-monty/crates/monty/test_cases/namedtuple__missing_attr.py +11 -0
  481. package/ref-monty/crates/monty/test_cases/namedtuple__ops.py +34 -0
  482. package/ref-monty/crates/monty/test_cases/nonlocal__error_module_level.py +3 -0
  483. package/ref-monty/crates/monty/test_cases/nonlocal__ops.py +353 -0
  484. package/ref-monty/crates/monty/test_cases/os__environ.py +40 -0
  485. package/ref-monty/crates/monty/test_cases/os__getenv_key_list_error.py +5 -0
  486. package/ref-monty/crates/monty/test_cases/os__getenv_key_type_error.py +5 -0
  487. package/ref-monty/crates/monty/test_cases/parse_error__complex.py +3 -0
  488. package/ref-monty/crates/monty/test_cases/pathlib__import.py +11 -0
  489. package/ref-monty/crates/monty/test_cases/pathlib__os.py +136 -0
  490. package/ref-monty/crates/monty/test_cases/pathlib__os_read_error.py +12 -0
  491. package/ref-monty/crates/monty/test_cases/pathlib__pure.py +81 -0
  492. package/ref-monty/crates/monty/test_cases/pyobject__cycle_dict_self.py +5 -0
  493. package/ref-monty/crates/monty/test_cases/pyobject__cycle_list_dict.py +6 -0
  494. package/ref-monty/crates/monty/test_cases/pyobject__cycle_list_self.py +5 -0
  495. package/ref-monty/crates/monty/test_cases/pyobject__cycle_multiple_refs.py +6 -0
  496. package/ref-monty/crates/monty/test_cases/range__error_no_args.py +2 -0
  497. package/ref-monty/crates/monty/test_cases/range__error_step_zero.py +2 -0
  498. package/ref-monty/crates/monty/test_cases/range__error_too_many_args.py +2 -0
  499. package/ref-monty/crates/monty/test_cases/range__getitem_index_error.py +10 -0
  500. package/ref-monty/crates/monty/test_cases/range__ops.py +236 -0
  501. package/ref-monty/crates/monty/test_cases/re__basic.py +756 -0
  502. package/ref-monty/crates/monty/test_cases/re__grouping.py +241 -0
  503. package/ref-monty/crates/monty/test_cases/re__match.py +148 -0
  504. package/ref-monty/crates/monty/test_cases/recursion__deep_drop.py +26 -0
  505. package/ref-monty/crates/monty/test_cases/recursion__deep_eq.py +23 -0
  506. package/ref-monty/crates/monty/test_cases/recursion__deep_hash.py +46 -0
  507. package/ref-monty/crates/monty/test_cases/recursion__deep_repr.py +12 -0
  508. package/ref-monty/crates/monty/test_cases/recursion__function_depth.py +13 -0
  509. package/ref-monty/crates/monty/test_cases/refcount__cycle_mutual_reference.py +18 -0
  510. package/ref-monty/crates/monty/test_cases/refcount__cycle_self_reference.py +12 -0
  511. package/ref-monty/crates/monty/test_cases/refcount__dict_basic.py +5 -0
  512. package/ref-monty/crates/monty/test_cases/refcount__dict_get.py +5 -0
  513. package/ref-monty/crates/monty/test_cases/refcount__dict_keys_and.py +14 -0
  514. package/ref-monty/crates/monty/test_cases/refcount__dict_overwrite.py +6 -0
  515. package/ref-monty/crates/monty/test_cases/refcount__gather_cleanup.py +16 -0
  516. package/ref-monty/crates/monty/test_cases/refcount__gather_exception.py +18 -0
  517. package/ref-monty/crates/monty/test_cases/refcount__gather_nested_cancel.py +25 -0
  518. package/ref-monty/crates/monty/test_cases/refcount__immediate_skipped.py +4 -0
  519. package/ref-monty/crates/monty/test_cases/refcount__kwargs_unpacking.py +27 -0
  520. package/ref-monty/crates/monty/test_cases/refcount__list_append_multiple.py +6 -0
  521. package/ref-monty/crates/monty/test_cases/refcount__list_append_ref.py +5 -0
  522. package/ref-monty/crates/monty/test_cases/refcount__list_concat.py +5 -0
  523. package/ref-monty/crates/monty/test_cases/refcount__list_getitem.py +5 -0
  524. package/ref-monty/crates/monty/test_cases/refcount__list_iadd.py +5 -0
  525. package/ref-monty/crates/monty/test_cases/refcount__nested_list.py +4 -0
  526. package/ref-monty/crates/monty/test_cases/refcount__re_pattern_sub_error_paths.py +37 -0
  527. package/ref-monty/crates/monty/test_cases/refcount__re_search_match.py +34 -0
  528. package/ref-monty/crates/monty/test_cases/refcount__re_sub_error_paths.py +31 -0
  529. package/ref-monty/crates/monty/test_cases/refcount__shared_reference.py +4 -0
  530. package/ref-monty/crates/monty/test_cases/refcount__single_list.py +3 -0
  531. package/ref-monty/crates/monty/test_cases/repr__cycle_detection.py +24 -0
  532. package/ref-monty/crates/monty/test_cases/set__ops.py +191 -0
  533. package/ref-monty/crates/monty/test_cases/set__review_bugs.py +35 -0
  534. package/ref-monty/crates/monty/test_cases/set__unpack_type_error.py +2 -0
  535. package/ref-monty/crates/monty/test_cases/slice__invalid_indices.py +2 -0
  536. package/ref-monty/crates/monty/test_cases/slice__kwargs.py +9 -0
  537. package/ref-monty/crates/monty/test_cases/slice__no_args.py +9 -0
  538. package/ref-monty/crates/monty/test_cases/slice__ops.py +149 -0
  539. package/ref-monty/crates/monty/test_cases/slice__step_zero.py +9 -0
  540. package/ref-monty/crates/monty/test_cases/slice__step_zero_bytes.py +9 -0
  541. package/ref-monty/crates/monty/test_cases/slice__step_zero_range.py +9 -0
  542. package/ref-monty/crates/monty/test_cases/slice__step_zero_str.py +9 -0
  543. package/ref-monty/crates/monty/test_cases/slice__step_zero_tuple.py +9 -0
  544. package/ref-monty/crates/monty/test_cases/slice__too_many_args.py +9 -0
  545. package/ref-monty/crates/monty/test_cases/str__getitem_index_error.py +10 -0
  546. package/ref-monty/crates/monty/test_cases/str__index_not_found.py +9 -0
  547. package/ref-monty/crates/monty/test_cases/str__join_no_args.py +9 -0
  548. package/ref-monty/crates/monty/test_cases/str__join_non_string.py +9 -0
  549. package/ref-monty/crates/monty/test_cases/str__join_not_iterable.py +9 -0
  550. package/ref-monty/crates/monty/test_cases/str__join_too_many_args.py +9 -0
  551. package/ref-monty/crates/monty/test_cases/str__methods.py +327 -0
  552. package/ref-monty/crates/monty/test_cases/str__ops.py +162 -0
  553. package/ref-monty/crates/monty/test_cases/str__partition_empty.py +9 -0
  554. package/ref-monty/crates/monty/test_cases/str__rsplit_empty_sep.py +9 -0
  555. package/ref-monty/crates/monty/test_cases/str__split_empty_sep.py +9 -0
  556. package/ref-monty/crates/monty/test_cases/sys__types.py +7 -0
  557. package/ref-monty/crates/monty/test_cases/traceback__division_error.py +30 -0
  558. package/ref-monty/crates/monty/test_cases/traceback__index_error.py +17 -0
  559. package/ref-monty/crates/monty/test_cases/traceback__insert_as_int.py +10 -0
  560. package/ref-monty/crates/monty/test_cases/traceback__nested_call.py +29 -0
  561. package/ref-monty/crates/monty/test_cases/traceback__nonlocal_module_scope.py +10 -0
  562. package/ref-monty/crates/monty/test_cases/traceback__nonlocal_unbound.py +24 -0
  563. package/ref-monty/crates/monty/test_cases/traceback__range_as_int.py +9 -0
  564. package/ref-monty/crates/monty/test_cases/traceback__recursion_error.py +23 -0
  565. package/ref-monty/crates/monty/test_cases/traceback__set_mutation.py +11 -0
  566. package/ref-monty/crates/monty/test_cases/traceback__undefined_attr_call.py +16 -0
  567. package/ref-monty/crates/monty/test_cases/traceback__undefined_call.py +16 -0
  568. package/ref-monty/crates/monty/test_cases/traceback__undefined_raise.py +16 -0
  569. package/ref-monty/crates/monty/test_cases/try_except__all.py +472 -0
  570. package/ref-monty/crates/monty/test_cases/try_except__bare_raise_no_context.py +2 -0
  571. package/ref-monty/crates/monty/test_cases/try_except__invalid_type.py +5 -0
  572. package/ref-monty/crates/monty/test_cases/tuple__getitem_out_of_bounds.py +3 -0
  573. package/ref-monty/crates/monty/test_cases/tuple__index_not_found.py +9 -0
  574. package/ref-monty/crates/monty/test_cases/tuple__index_start_gt_end.py +10 -0
  575. package/ref-monty/crates/monty/test_cases/tuple__methods.py +19 -0
  576. package/ref-monty/crates/monty/test_cases/tuple__ops.py +133 -0
  577. package/ref-monty/crates/monty/test_cases/tuple__unpack_type_error.py +2 -0
  578. package/ref-monty/crates/monty/test_cases/type__builtin_attr_error.py +9 -0
  579. package/ref-monty/crates/monty/test_cases/type__bytes_negative.py +2 -0
  580. package/ref-monty/crates/monty/test_cases/type__cell_not_builtin.py +9 -0
  581. package/ref-monty/crates/monty/test_cases/type__exception_attr_error.py +11 -0
  582. package/ref-monty/crates/monty/test_cases/type__float_conversion_error.py +2 -0
  583. package/ref-monty/crates/monty/test_cases/type__float_repr_both_quotes.py +9 -0
  584. package/ref-monty/crates/monty/test_cases/type__float_repr_newline.py +9 -0
  585. package/ref-monty/crates/monty/test_cases/type__float_repr_single_quote.py +9 -0
  586. package/ref-monty/crates/monty/test_cases/type__int_conversion_error.py +2 -0
  587. package/ref-monty/crates/monty/test_cases/type__list_not_iterable.py +2 -0
  588. package/ref-monty/crates/monty/test_cases/type__non_builtin_name_error.py +9 -0
  589. package/ref-monty/crates/monty/test_cases/type__ops.py +200 -0
  590. package/ref-monty/crates/monty/test_cases/type__shadow_exc.py +3 -0
  591. package/ref-monty/crates/monty/test_cases/type__shadow_int.py +9 -0
  592. package/ref-monty/crates/monty/test_cases/type__shadow_len.py +3 -0
  593. package/ref-monty/crates/monty/test_cases/type__tuple_not_iterable.py +2 -0
  594. package/ref-monty/crates/monty/test_cases/type_error__int_add_list.py +2 -0
  595. package/ref-monty/crates/monty/test_cases/type_error__int_div_str.py +2 -0
  596. package/ref-monty/crates/monty/test_cases/type_error__int_floordiv_str.py +2 -0
  597. package/ref-monty/crates/monty/test_cases/type_error__int_iadd_str.py +3 -0
  598. package/ref-monty/crates/monty/test_cases/type_error__int_mod_str.py +2 -0
  599. package/ref-monty/crates/monty/test_cases/type_error__int_pow_str.py +2 -0
  600. package/ref-monty/crates/monty/test_cases/type_error__int_sub_str.py +2 -0
  601. package/ref-monty/crates/monty/test_cases/type_error__list_add_int.py +2 -0
  602. package/ref-monty/crates/monty/test_cases/type_error__list_add_str.py +2 -0
  603. package/ref-monty/crates/monty/test_cases/type_error__list_iadd_int.py +6 -0
  604. package/ref-monty/crates/monty/test_cases/type_error__str_add_int.py +2 -0
  605. package/ref-monty/crates/monty/test_cases/type_error__str_iadd_int.py +3 -0
  606. package/ref-monty/crates/monty/test_cases/type_error__unary_invert_str.py +3 -0
  607. package/ref-monty/crates/monty/test_cases/type_error__unary_minus_str.py +4 -0
  608. package/ref-monty/crates/monty/test_cases/type_error__unary_neg_str.py +3 -0
  609. package/ref-monty/crates/monty/test_cases/type_error__unary_plus_str.py +4 -0
  610. package/ref-monty/crates/monty/test_cases/typing__types.py +24 -0
  611. package/ref-monty/crates/monty/test_cases/unpack__nested.py +48 -0
  612. package/ref-monty/crates/monty/test_cases/unpack__non_sequence.py +9 -0
  613. package/ref-monty/crates/monty/test_cases/unpack__not_enough.py +9 -0
  614. package/ref-monty/crates/monty/test_cases/unpack__ops.py +153 -0
  615. package/ref-monty/crates/monty/test_cases/unpack__star_not_enough.py +9 -0
  616. package/ref-monty/crates/monty/test_cases/unpack__too_many.py +9 -0
  617. package/ref-monty/crates/monty/test_cases/version__cpython.py +4 -0
  618. package/ref-monty/crates/monty/test_cases/walrus__all.py +178 -0
  619. package/ref-monty/crates/monty/test_cases/while__all.py +206 -0
  620. package/ref-monty/crates/monty/tests/asyncio.rs +764 -0
  621. package/ref-monty/crates/monty/tests/binary_serde.rs +185 -0
  622. package/ref-monty/crates/monty/tests/bytecode_limits.rs +248 -0
  623. package/ref-monty/crates/monty/tests/datatest_runner.rs +2029 -0
  624. package/ref-monty/crates/monty/tests/inputs.rs +420 -0
  625. package/ref-monty/crates/monty/tests/json_serde.rs +250 -0
  626. package/ref-monty/crates/monty/tests/main.rs +71 -0
  627. package/ref-monty/crates/monty/tests/math_module.rs +114 -0
  628. package/ref-monty/crates/monty/tests/name_lookup.rs +482 -0
  629. package/ref-monty/crates/monty/tests/os_tests.rs +459 -0
  630. package/ref-monty/crates/monty/tests/parse_errors.rs +441 -0
  631. package/ref-monty/crates/monty/tests/print_writer.rs +238 -0
  632. package/ref-monty/crates/monty/tests/py_object.rs +121 -0
  633. package/ref-monty/crates/monty/tests/regex.rs +90 -0
  634. package/ref-monty/crates/monty/tests/repl.rs +344 -0
  635. package/ref-monty/crates/monty/tests/resource_limits.rs +1826 -0
  636. package/ref-monty/crates/monty/tests/try_from.rs +167 -0
  637. package/ref-monty/crates/monty-cli/Cargo.toml +25 -0
  638. package/ref-monty/crates/monty-cli/src/main.rs +541 -0
  639. package/ref-monty/crates/monty-js/.cargo/config.toml +2 -0
  640. package/ref-monty/crates/monty-js/.prettierignore +8 -0
  641. package/ref-monty/crates/monty-js/Cargo.toml +32 -0
  642. package/ref-monty/crates/monty-js/README.md +207 -0
  643. package/ref-monty/crates/monty-js/__test__/async.spec.ts +350 -0
  644. package/ref-monty/crates/monty-js/__test__/basic.spec.ts +114 -0
  645. package/ref-monty/crates/monty-js/__test__/exceptions.spec.ts +427 -0
  646. package/ref-monty/crates/monty-js/__test__/external.spec.ts +354 -0
  647. package/ref-monty/crates/monty-js/__test__/inputs.spec.ts +143 -0
  648. package/ref-monty/crates/monty-js/__test__/limits.spec.ts +162 -0
  649. package/ref-monty/crates/monty-js/__test__/package.json +3 -0
  650. package/ref-monty/crates/monty-js/__test__/print.spec.ts +229 -0
  651. package/ref-monty/crates/monty-js/__test__/repl.spec.ts +34 -0
  652. package/ref-monty/crates/monty-js/__test__/serialize.spec.ts +205 -0
  653. package/ref-monty/crates/monty-js/__test__/start.spec.ts +443 -0
  654. package/ref-monty/crates/monty-js/__test__/type_check.spec.ts +147 -0
  655. package/ref-monty/crates/monty-js/__test__/types.spec.ts +319 -0
  656. package/ref-monty/crates/monty-js/build.rs +61 -0
  657. package/ref-monty/crates/monty-js/index-header.d.ts +3 -0
  658. package/ref-monty/crates/monty-js/package-lock.json +4694 -0
  659. package/ref-monty/crates/monty-js/package.json +100 -0
  660. package/ref-monty/crates/monty-js/scripts/smoke-test.sh +69 -0
  661. package/ref-monty/crates/monty-js/smoke-test/package.json +17 -0
  662. package/ref-monty/crates/monty-js/smoke-test/test.ts +171 -0
  663. package/ref-monty/crates/monty-js/smoke-test/tsconfig.json +11 -0
  664. package/ref-monty/crates/monty-js/src/convert.rs +648 -0
  665. package/ref-monty/crates/monty-js/src/exceptions.rs +293 -0
  666. package/ref-monty/crates/monty-js/src/lib.rs +41 -0
  667. package/ref-monty/crates/monty-js/src/limits.rs +53 -0
  668. package/ref-monty/crates/monty-js/src/monty_cls.rs +1407 -0
  669. package/ref-monty/crates/monty-js/tsconfig.json +17 -0
  670. package/ref-monty/crates/monty-js/wrapper.ts +701 -0
  671. package/ref-monty/crates/monty-python/Cargo.toml +38 -0
  672. package/ref-monty/crates/monty-python/README.md +134 -0
  673. package/ref-monty/crates/monty-python/build.rs +4 -0
  674. package/ref-monty/crates/monty-python/example.py +40 -0
  675. package/ref-monty/crates/monty-python/exercise.py +46 -0
  676. package/ref-monty/crates/monty-python/pyproject.toml +57 -0
  677. package/ref-monty/crates/monty-python/python/pydantic_monty/__init__.py +281 -0
  678. package/ref-monty/crates/monty-python/python/pydantic_monty/_monty.pyi +677 -0
  679. package/ref-monty/crates/monty-python/python/pydantic_monty/os_access.py +933 -0
  680. package/ref-monty/crates/monty-python/python/pydantic_monty/py.typed +0 -0
  681. package/ref-monty/crates/monty-python/src/convert.rs +273 -0
  682. package/ref-monty/crates/monty-python/src/dataclass.rs +461 -0
  683. package/ref-monty/crates/monty-python/src/exceptions.rs +557 -0
  684. package/ref-monty/crates/monty-python/src/external.rs +165 -0
  685. package/ref-monty/crates/monty-python/src/lib.rs +77 -0
  686. package/ref-monty/crates/monty-python/src/limits.rs +142 -0
  687. package/ref-monty/crates/monty-python/src/monty_cls.rs +1650 -0
  688. package/ref-monty/crates/monty-python/src/repl.rs +470 -0
  689. package/ref-monty/crates/monty-python/src/serialization.rs +761 -0
  690. package/ref-monty/crates/monty-python/tests/test_async.py +1201 -0
  691. package/ref-monty/crates/monty-python/tests/test_basic.py +66 -0
  692. package/ref-monty/crates/monty-python/tests/test_dataclasses.py +971 -0
  693. package/ref-monty/crates/monty-python/tests/test_exceptions.py +361 -0
  694. package/ref-monty/crates/monty-python/tests/test_external.py +367 -0
  695. package/ref-monty/crates/monty-python/tests/test_inputs.py +126 -0
  696. package/ref-monty/crates/monty-python/tests/test_limits.py +257 -0
  697. package/ref-monty/crates/monty-python/tests/test_os_access.py +1286 -0
  698. package/ref-monty/crates/monty-python/tests/test_os_access_compat.py +731 -0
  699. package/ref-monty/crates/monty-python/tests/test_os_access_raw.py +483 -0
  700. package/ref-monty/crates/monty-python/tests/test_os_calls.py +819 -0
  701. package/ref-monty/crates/monty-python/tests/test_print.py +208 -0
  702. package/ref-monty/crates/monty-python/tests/test_re.py +170 -0
  703. package/ref-monty/crates/monty-python/tests/test_readme_examples.py +20 -0
  704. package/ref-monty/crates/monty-python/tests/test_repl.py +749 -0
  705. package/ref-monty/crates/monty-python/tests/test_serialize.py +284 -0
  706. package/ref-monty/crates/monty-python/tests/test_start.py +346 -0
  707. package/ref-monty/crates/monty-python/tests/test_threading.py +163 -0
  708. package/ref-monty/crates/monty-python/tests/test_type_check.py +344 -0
  709. package/ref-monty/crates/monty-python/tests/test_types.py +553 -0
  710. package/ref-monty/crates/monty-type-checking/Cargo.toml +32 -0
  711. package/ref-monty/crates/monty-type-checking/src/db.rs +116 -0
  712. package/ref-monty/crates/monty-type-checking/src/lib.rs +4 -0
  713. package/ref-monty/crates/monty-type-checking/src/type_check.rs +280 -0
  714. package/ref-monty/crates/monty-type-checking/tests/bad_types.py +109 -0
  715. package/ref-monty/crates/monty-type-checking/tests/bad_types_output.txt +21 -0
  716. package/ref-monty/crates/monty-type-checking/tests/good_types.py +475 -0
  717. package/ref-monty/crates/monty-type-checking/tests/main.rs +205 -0
  718. package/ref-monty/crates/monty-type-checking/tests/reveal_types.py +56 -0
  719. package/ref-monty/crates/monty-type-checking/tests/reveal_types_output.txt +41 -0
  720. package/ref-monty/crates/monty-typeshed/Cargo.toml +29 -0
  721. package/ref-monty/crates/monty-typeshed/README.md +11 -0
  722. package/ref-monty/crates/monty-typeshed/build.rs +101 -0
  723. package/ref-monty/crates/monty-typeshed/custom/README.md +1 -0
  724. package/ref-monty/crates/monty-typeshed/custom/asyncio.pyi +138 -0
  725. package/ref-monty/crates/monty-typeshed/custom/os.pyi +87 -0
  726. package/ref-monty/crates/monty-typeshed/custom/sys.pyi +33 -0
  727. package/ref-monty/crates/monty-typeshed/src/lib.rs +56 -0
  728. package/ref-monty/crates/monty-typeshed/update.py +321 -0
  729. package/ref-monty/crates/monty-typeshed/vendor/typeshed/source_commit.txt +1 -0
  730. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/VERSIONS +20 -0
  731. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/_collections_abc.pyi +105 -0
  732. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/_typeshed/__init__.pyi +394 -0
  733. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/asyncio.pyi +138 -0
  734. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/builtins.pyi +1434 -0
  735. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/collections/__init__.pyi +527 -0
  736. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/collections/abc.pyi +2 -0
  737. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/dataclasses.pyi +502 -0
  738. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/enum.pyi +376 -0
  739. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/math.pyi +149 -0
  740. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/os.pyi +87 -0
  741. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/pathlib/__init__.pyi +395 -0
  742. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/pathlib/types.pyi +8 -0
  743. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/re.pyi +337 -0
  744. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/sys.pyi +33 -0
  745. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/types.pyi +741 -0
  746. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/typing.pyi +1217 -0
  747. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/typing_extensions.pyi +716 -0
  748. package/ref-monty/docs/usage-guide.md +117 -0
  749. package/ref-monty/examples/README.md +3 -0
  750. package/ref-monty/examples/expense_analysis/README.md +3 -0
  751. package/ref-monty/examples/expense_analysis/data.py +124 -0
  752. package/ref-monty/examples/expense_analysis/main.py +115 -0
  753. package/ref-monty/examples/sql_playground/README.md +20 -0
  754. package/ref-monty/examples/sql_playground/external_functions.py +129 -0
  755. package/ref-monty/examples/sql_playground/main.py +81 -0
  756. package/ref-monty/examples/sql_playground/sandbox_code.py +82 -0
  757. package/ref-monty/examples/sql_playground/type_stubs.pyi +14 -0
  758. package/ref-monty/examples/web_scraper/README.md +15 -0
  759. package/ref-monty/examples/web_scraper/browser.py +56 -0
  760. package/ref-monty/examples/web_scraper/example_code.py +59 -0
  761. package/ref-monty/examples/web_scraper/external_functions.py +324 -0
  762. package/ref-monty/examples/web_scraper/main.py +193 -0
  763. package/ref-monty/examples/web_scraper/sub_agent.py +79 -0
  764. package/ref-monty/monty-npm.md +235 -0
  765. package/ref-monty/pyproject.toml +162 -0
  766. package/ref-monty/scripts/check_imports.py +91 -0
  767. package/ref-monty/scripts/codecov_diff.py +412 -0
  768. package/ref-monty/scripts/complete_tests.py +146 -0
  769. package/ref-monty/scripts/flamegraph_to_text.py +208 -0
  770. package/ref-monty/scripts/iter_test_methods.py +540 -0
  771. package/ref-monty/scripts/run_traceback.py +180 -0
  772. package/ref-monty/scripts/startup_performance.py +130 -0
  773. package/ref-monty/uv.lock +1779 -0
  774. package/temp_resend_cli/repo/.github/scripts/pr-title-check.js +34 -0
  775. package/temp_resend_cli/repo/.github/workflows/ci.yml +67 -0
  776. package/temp_resend_cli/repo/.github/workflows/post-release.yml +51 -0
  777. package/temp_resend_cli/repo/.github/workflows/pr-title-check.yml +13 -0
  778. package/temp_resend_cli/repo/.github/workflows/release.yml +175 -0
  779. package/temp_resend_cli/repo/.github/workflows/test-install-unix.yml +34 -0
  780. package/temp_resend_cli/repo/.github/workflows/test-install-windows.yml +48 -0
  781. package/temp_resend_cli/repo/CHANGELOG.md +31 -0
  782. package/temp_resend_cli/repo/LICENSE +21 -0
  783. package/temp_resend_cli/repo/README.md +450 -0
  784. package/temp_resend_cli/repo/biome.json +36 -0
  785. package/temp_resend_cli/repo/install.ps1 +141 -0
  786. package/temp_resend_cli/repo/install.sh +301 -0
  787. package/temp_resend_cli/repo/package.json +61 -0
  788. package/temp_resend_cli/repo/pnpm-lock.yaml +2439 -0
  789. package/temp_resend_cli/repo/renovate.json +4 -0
  790. package/temp_resend_cli/repo/src/cli.ts +98 -0
  791. package/temp_resend_cli/repo/src/commands/api-keys/create.ts +114 -0
  792. package/temp_resend_cli/repo/src/commands/api-keys/delete.ts +47 -0
  793. package/temp_resend_cli/repo/src/commands/api-keys/index.ts +26 -0
  794. package/temp_resend_cli/repo/src/commands/api-keys/list.ts +35 -0
  795. package/temp_resend_cli/repo/src/commands/api-keys/utils.ts +8 -0
  796. package/temp_resend_cli/repo/src/commands/auth/index.ts +20 -0
  797. package/temp_resend_cli/repo/src/commands/auth/login.ts +234 -0
  798. package/temp_resend_cli/repo/src/commands/auth/logout.ts +105 -0
  799. package/temp_resend_cli/repo/src/commands/broadcasts/create.ts +196 -0
  800. package/temp_resend_cli/repo/src/commands/broadcasts/delete.ts +46 -0
  801. package/temp_resend_cli/repo/src/commands/broadcasts/get.ts +59 -0
  802. package/temp_resend_cli/repo/src/commands/broadcasts/index.ts +43 -0
  803. package/temp_resend_cli/repo/src/commands/broadcasts/list.ts +60 -0
  804. package/temp_resend_cli/repo/src/commands/broadcasts/send.ts +56 -0
  805. package/temp_resend_cli/repo/src/commands/broadcasts/update.ts +95 -0
  806. package/temp_resend_cli/repo/src/commands/broadcasts/utils.ts +35 -0
  807. package/temp_resend_cli/repo/src/commands/contact-properties/create.ts +118 -0
  808. package/temp_resend_cli/repo/src/commands/contact-properties/delete.ts +48 -0
  809. package/temp_resend_cli/repo/src/commands/contact-properties/get.ts +46 -0
  810. package/temp_resend_cli/repo/src/commands/contact-properties/index.ts +48 -0
  811. package/temp_resend_cli/repo/src/commands/contact-properties/list.ts +68 -0
  812. package/temp_resend_cli/repo/src/commands/contact-properties/update.ts +88 -0
  813. package/temp_resend_cli/repo/src/commands/contact-properties/utils.ts +17 -0
  814. package/temp_resend_cli/repo/src/commands/contacts/add-segment.ts +78 -0
  815. package/temp_resend_cli/repo/src/commands/contacts/create.ts +122 -0
  816. package/temp_resend_cli/repo/src/commands/contacts/delete.ts +49 -0
  817. package/temp_resend_cli/repo/src/commands/contacts/get.ts +53 -0
  818. package/temp_resend_cli/repo/src/commands/contacts/index.ts +58 -0
  819. package/temp_resend_cli/repo/src/commands/contacts/list.ts +57 -0
  820. package/temp_resend_cli/repo/src/commands/contacts/remove-segment.ts +48 -0
  821. package/temp_resend_cli/repo/src/commands/contacts/segments.ts +39 -0
  822. package/temp_resend_cli/repo/src/commands/contacts/topics.ts +45 -0
  823. package/temp_resend_cli/repo/src/commands/contacts/update-topics.ts +90 -0
  824. package/temp_resend_cli/repo/src/commands/contacts/update.ts +77 -0
  825. package/temp_resend_cli/repo/src/commands/contacts/utils.ts +119 -0
  826. package/temp_resend_cli/repo/src/commands/doctor.ts +216 -0
  827. package/temp_resend_cli/repo/src/commands/domains/create.ts +83 -0
  828. package/temp_resend_cli/repo/src/commands/domains/delete.ts +42 -0
  829. package/temp_resend_cli/repo/src/commands/domains/get.ts +47 -0
  830. package/temp_resend_cli/repo/src/commands/domains/index.ts +35 -0
  831. package/temp_resend_cli/repo/src/commands/domains/list.ts +53 -0
  832. package/temp_resend_cli/repo/src/commands/domains/update.ts +75 -0
  833. package/temp_resend_cli/repo/src/commands/domains/utils.ts +44 -0
  834. package/temp_resend_cli/repo/src/commands/domains/verify.ts +38 -0
  835. package/temp_resend_cli/repo/src/commands/emails/batch.ts +140 -0
  836. package/temp_resend_cli/repo/src/commands/emails/get.ts +44 -0
  837. package/temp_resend_cli/repo/src/commands/emails/index.ts +30 -0
  838. package/temp_resend_cli/repo/src/commands/emails/list.ts +84 -0
  839. package/temp_resend_cli/repo/src/commands/emails/receiving/attachment.ts +55 -0
  840. package/temp_resend_cli/repo/src/commands/emails/receiving/attachments.ts +68 -0
  841. package/temp_resend_cli/repo/src/commands/emails/receiving/get.ts +58 -0
  842. package/temp_resend_cli/repo/src/commands/emails/receiving/index.ts +28 -0
  843. package/temp_resend_cli/repo/src/commands/emails/receiving/list.ts +59 -0
  844. package/temp_resend_cli/repo/src/commands/emails/receiving/utils.ts +38 -0
  845. package/temp_resend_cli/repo/src/commands/emails/send.ts +189 -0
  846. package/temp_resend_cli/repo/src/commands/open.ts +27 -0
  847. package/temp_resend_cli/repo/src/commands/segments/create.ts +50 -0
  848. package/temp_resend_cli/repo/src/commands/segments/delete.ts +47 -0
  849. package/temp_resend_cli/repo/src/commands/segments/get.ts +38 -0
  850. package/temp_resend_cli/repo/src/commands/segments/index.ts +36 -0
  851. package/temp_resend_cli/repo/src/commands/segments/list.ts +58 -0
  852. package/temp_resend_cli/repo/src/commands/segments/utils.ts +7 -0
  853. package/temp_resend_cli/repo/src/commands/teams/index.ts +10 -0
  854. package/temp_resend_cli/repo/src/commands/teams/list.ts +35 -0
  855. package/temp_resend_cli/repo/src/commands/teams/remove.ts +86 -0
  856. package/temp_resend_cli/repo/src/commands/teams/switch.ts +76 -0
  857. package/temp_resend_cli/repo/src/commands/topics/create.ts +73 -0
  858. package/temp_resend_cli/repo/src/commands/topics/delete.ts +47 -0
  859. package/temp_resend_cli/repo/src/commands/topics/get.ts +42 -0
  860. package/temp_resend_cli/repo/src/commands/topics/index.ts +42 -0
  861. package/temp_resend_cli/repo/src/commands/topics/list.ts +34 -0
  862. package/temp_resend_cli/repo/src/commands/topics/update.ts +59 -0
  863. package/temp_resend_cli/repo/src/commands/topics/utils.ts +16 -0
  864. package/temp_resend_cli/repo/src/commands/webhooks/create.ts +128 -0
  865. package/temp_resend_cli/repo/src/commands/webhooks/delete.ts +49 -0
  866. package/temp_resend_cli/repo/src/commands/webhooks/get.ts +42 -0
  867. package/temp_resend_cli/repo/src/commands/webhooks/index.ts +42 -0
  868. package/temp_resend_cli/repo/src/commands/webhooks/list.ts +55 -0
  869. package/temp_resend_cli/repo/src/commands/webhooks/listen.ts +379 -0
  870. package/temp_resend_cli/repo/src/commands/webhooks/update.ts +83 -0
  871. package/temp_resend_cli/repo/src/commands/webhooks/utils.ts +36 -0
  872. package/temp_resend_cli/repo/src/commands/whoami.ts +71 -0
  873. package/temp_resend_cli/repo/src/lib/actions.ts +157 -0
  874. package/temp_resend_cli/repo/src/lib/client.ts +37 -0
  875. package/temp_resend_cli/repo/src/lib/config.ts +217 -0
  876. package/temp_resend_cli/repo/src/lib/files.ts +15 -0
  877. package/temp_resend_cli/repo/src/lib/help-text.ts +38 -0
  878. package/temp_resend_cli/repo/src/lib/output.ts +56 -0
  879. package/temp_resend_cli/repo/src/lib/pagination.ts +36 -0
  880. package/temp_resend_cli/repo/src/lib/prompts.ts +149 -0
  881. package/temp_resend_cli/repo/src/lib/spinner.ts +100 -0
  882. package/temp_resend_cli/repo/src/lib/table.ts +57 -0
  883. package/temp_resend_cli/repo/src/lib/tty.ts +28 -0
  884. package/temp_resend_cli/repo/src/lib/update-check.ts +169 -0
  885. package/temp_resend_cli/repo/src/lib/version.ts +4 -0
  886. package/temp_resend_cli/repo/tests/commands/api-keys/create.test.ts +196 -0
  887. package/temp_resend_cli/repo/tests/commands/api-keys/delete.test.ts +157 -0
  888. package/temp_resend_cli/repo/tests/commands/api-keys/list.test.ts +134 -0
  889. package/temp_resend_cli/repo/tests/commands/auth/login.test.ts +153 -0
  890. package/temp_resend_cli/repo/tests/commands/auth/logout.test.ts +153 -0
  891. package/temp_resend_cli/repo/tests/commands/broadcasts/create.test.ts +454 -0
  892. package/temp_resend_cli/repo/tests/commands/broadcasts/delete.test.ts +183 -0
  893. package/temp_resend_cli/repo/tests/commands/broadcasts/get.test.ts +147 -0
  894. package/temp_resend_cli/repo/tests/commands/broadcasts/list.test.ts +199 -0
  895. package/temp_resend_cli/repo/tests/commands/broadcasts/send.test.ts +162 -0
  896. package/temp_resend_cli/repo/tests/commands/broadcasts/update.test.ts +288 -0
  897. package/temp_resend_cli/repo/tests/commands/contact-properties/create.test.ts +251 -0
  898. package/temp_resend_cli/repo/tests/commands/contact-properties/delete.test.ts +184 -0
  899. package/temp_resend_cli/repo/tests/commands/contact-properties/get.test.ts +145 -0
  900. package/temp_resend_cli/repo/tests/commands/contact-properties/list.test.ts +181 -0
  901. package/temp_resend_cli/repo/tests/commands/contact-properties/update.test.ts +217 -0
  902. package/temp_resend_cli/repo/tests/commands/contacts/add-segment.test.ts +189 -0
  903. package/temp_resend_cli/repo/tests/commands/contacts/create.test.ts +271 -0
  904. package/temp_resend_cli/repo/tests/commands/contacts/delete.test.ts +193 -0
  905. package/temp_resend_cli/repo/tests/commands/contacts/get.test.ts +149 -0
  906. package/temp_resend_cli/repo/tests/commands/contacts/list.test.ts +176 -0
  907. package/temp_resend_cli/repo/tests/commands/contacts/remove-segment.test.ts +167 -0
  908. package/temp_resend_cli/repo/tests/commands/contacts/segments.test.ts +168 -0
  909. package/temp_resend_cli/repo/tests/commands/contacts/topics.test.ts +164 -0
  910. package/temp_resend_cli/repo/tests/commands/contacts/update-topics.test.ts +248 -0
  911. package/temp_resend_cli/repo/tests/commands/contacts/update.test.ts +206 -0
  912. package/temp_resend_cli/repo/tests/commands/doctor.test.ts +164 -0
  913. package/temp_resend_cli/repo/tests/commands/domains/create.test.ts +193 -0
  914. package/temp_resend_cli/repo/tests/commands/domains/delete.test.ts +157 -0
  915. package/temp_resend_cli/repo/tests/commands/domains/get.test.ts +138 -0
  916. package/temp_resend_cli/repo/tests/commands/domains/list.test.ts +165 -0
  917. package/temp_resend_cli/repo/tests/commands/domains/update.test.ts +224 -0
  918. package/temp_resend_cli/repo/tests/commands/domains/verify.test.ts +118 -0
  919. package/temp_resend_cli/repo/tests/commands/emails/batch.test.ts +324 -0
  920. package/temp_resend_cli/repo/tests/commands/emails/get.test.ts +132 -0
  921. package/temp_resend_cli/repo/tests/commands/emails/receiving/attachment.test.ts +141 -0
  922. package/temp_resend_cli/repo/tests/commands/emails/receiving/attachments.test.ts +169 -0
  923. package/temp_resend_cli/repo/tests/commands/emails/receiving/get.test.ts +141 -0
  924. package/temp_resend_cli/repo/tests/commands/emails/receiving/list.test.ts +182 -0
  925. package/temp_resend_cli/repo/tests/commands/emails/send.test.ts +312 -0
  926. package/temp_resend_cli/repo/tests/commands/segments/create.test.ts +164 -0
  927. package/temp_resend_cli/repo/tests/commands/segments/delete.test.ts +183 -0
  928. package/temp_resend_cli/repo/tests/commands/segments/get.test.ts +138 -0
  929. package/temp_resend_cli/repo/tests/commands/segments/list.test.ts +174 -0
  930. package/temp_resend_cli/repo/tests/commands/teams/list.test.ts +62 -0
  931. package/temp_resend_cli/repo/tests/commands/teams/remove.test.ts +110 -0
  932. package/temp_resend_cli/repo/tests/commands/teams/switch.test.ts +103 -0
  933. package/temp_resend_cli/repo/tests/commands/topics/create.test.ts +192 -0
  934. package/temp_resend_cli/repo/tests/commands/topics/delete.test.ts +157 -0
  935. package/temp_resend_cli/repo/tests/commands/topics/get.test.ts +126 -0
  936. package/temp_resend_cli/repo/tests/commands/topics/list.test.ts +125 -0
  937. package/temp_resend_cli/repo/tests/commands/topics/update.test.ts +178 -0
  938. package/temp_resend_cli/repo/tests/commands/webhooks/create.test.ts +225 -0
  939. package/temp_resend_cli/repo/tests/commands/webhooks/delete.test.ts +157 -0
  940. package/temp_resend_cli/repo/tests/commands/webhooks/get.test.ts +126 -0
  941. package/temp_resend_cli/repo/tests/commands/webhooks/list.test.ts +178 -0
  942. package/temp_resend_cli/repo/tests/commands/webhooks/update.test.ts +207 -0
  943. package/temp_resend_cli/repo/tests/commands/whoami.test.ts +98 -0
  944. package/temp_resend_cli/repo/tests/e2e/smoke.test.ts +93 -0
  945. package/temp_resend_cli/repo/tests/helpers.ts +86 -0
  946. package/temp_resend_cli/repo/tests/lib/client.test.ts +71 -0
  947. package/temp_resend_cli/repo/tests/lib/config.test.ts +451 -0
  948. package/temp_resend_cli/repo/tests/lib/files.test.ts +73 -0
  949. package/temp_resend_cli/repo/tests/lib/help-text.test.ts +97 -0
  950. package/temp_resend_cli/repo/tests/lib/output.test.ts +136 -0
  951. package/temp_resend_cli/repo/tests/lib/prompts.test.ts +185 -0
  952. package/temp_resend_cli/repo/tests/lib/spinner.test.ts +166 -0
  953. package/temp_resend_cli/repo/tests/lib/table.test.ts +63 -0
  954. package/temp_resend_cli/repo/tests/lib/tty.test.ts +89 -0
  955. package/temp_resend_cli/repo/tests/lib/update-check.test.ts +179 -0
  956. package/temp_resend_cli/repo/tsconfig.json +14 -0
  957. package/temp_resend_cli/repo/vitest.config.e2e.ts +8 -0
  958. package/temp_resend_cli/repo/vitest.config.ts +10 -0
  959. package/tests/test-mcp-browser-use-smoke.sh +28 -56
  960. package/tests/test-monty-smoke.sh +32 -0
  961. package/tests/test-resend-smoke.sh +36 -0
@@ -0,0 +1,1453 @@
1
+ //! Implementation of Python's `math` module.
2
+ //!
3
+ //! Provides mathematical functions and constants matching CPython 3.14 behavior
4
+ //! and error messages. All functions are pure computations that don't require
5
+ //! host involvement, so they return `Value` directly rather than `AttrCallResult`.
6
+ //!
7
+ //! ## Implemented functions
8
+ //!
9
+ //! **Rounding**: `floor`, `ceil`, `trunc`
10
+ //! **Roots & powers**: `sqrt`, `isqrt`, `cbrt`, `pow`, `exp`, `exp2`, `expm1`
11
+ //! **Logarithms**: `log`, `log2`, `log10`, `log1p`
12
+ //! **Trigonometric**: `sin`, `cos`, `tan`, `asin`, `acos`, `atan`, `atan2`
13
+ //! **Hyperbolic**: `sinh`, `cosh`, `tanh`, `asinh`, `acosh`, `atanh`
14
+ //! **Angular**: `degrees`, `radians`
15
+ //! **Float properties**: `fabs`, `isnan`, `isinf`, `isfinite`, `copysign`, `isclose`,
16
+ //! `nextafter`, `ulp`
17
+ //! **Integer math**: `factorial`, `gcd`, `lcm`, `comb`, `perm`
18
+ //! **Modular**: `fmod`, `remainder`, `modf`, `frexp`, `ldexp`
19
+ //! **Special**: `gamma`, `lgamma`, `erf`, `erfc`
20
+ //!
21
+ //! ## Constants
22
+ //!
23
+ //! `pi`, `e`, `tau`, `inf`, `nan`
24
+
25
+ use num_bigint::BigInt;
26
+ use smallvec::smallvec;
27
+
28
+ use crate::{
29
+ args::ArgValues,
30
+ bytecode::VM,
31
+ defer_drop, defer_drop_mut,
32
+ exception_private::{ExcType, RunResult, SimpleException},
33
+ heap::{Heap, HeapData, HeapId},
34
+ intern::{Interns, StaticStrings},
35
+ modules::ModuleFunctions,
36
+ resource::{ResourceError, ResourceTracker},
37
+ types::{LongInt, Module, PyTrait, allocate_tuple},
38
+ value::Value,
39
+ };
40
+
41
+ // ==========================
42
+ // Shared constants and error helpers
43
+ // ==========================
44
+
45
+ /// Returns a `ValueError` with the standard CPython "math domain error" message.
46
+ fn math_domain_error() -> crate::exception_private::RunError {
47
+ SimpleException::new_msg(ExcType::ValueError, "math domain error").into()
48
+ }
49
+
50
+ /// Returns an `OverflowError` with the standard CPython "math range error" message.
51
+ fn math_range_error() -> crate::exception_private::RunError {
52
+ SimpleException::new_msg(ExcType::OverflowError, "math range error").into()
53
+ }
54
+
55
+ /// Checks whether a computation overflowed (finite input produced infinite result).
56
+ ///
57
+ /// Returns `Err(OverflowError("math range error"))` if `result` is infinite
58
+ /// but `input` was finite.
59
+ fn check_range_error(result: f64, input: f64) -> RunResult<()> {
60
+ if result.is_infinite() && input.is_finite() {
61
+ Err(math_range_error())
62
+ } else {
63
+ Ok(())
64
+ }
65
+ }
66
+
67
+ /// Checks that a value is in the `[-1, 1]` range, raising `ValueError` if not.
68
+ ///
69
+ /// NaN passes through (it will propagate through the subsequent math operation).
70
+ /// Used by `math.asin` and `math.acos`.
71
+ fn require_unit_range(f: f64) -> RunResult<()> {
72
+ if !f.is_nan() && !(-1.0..=1.0).contains(&f) {
73
+ Err(SimpleException::new_msg(
74
+ ExcType::ValueError,
75
+ format!("expected a number in range from -1 up to 1, got {f:?}"),
76
+ )
77
+ .into())
78
+ } else {
79
+ Ok(())
80
+ }
81
+ }
82
+
83
+ /// Checks for non-positive integer arguments (poles of the Gamma function).
84
+ ///
85
+ /// These are the finite non-positive integers where Gamma diverges to ±∞.
86
+ /// Does NOT reject `-inf` — callers that need to reject it (like `math.gamma`)
87
+ /// must do so separately, since `lgamma(-inf)` is valid and returns `inf`.
88
+ #[expect(
89
+ clippy::float_cmp,
90
+ reason = "exact comparison detects integer poles of gamma function"
91
+ )]
92
+ fn check_gamma_pole(f: f64) -> RunResult<()> {
93
+ if f <= 0.0 && f == f.floor() && f.is_finite() {
94
+ Err(SimpleException::new_msg(
95
+ ExcType::ValueError,
96
+ format!("expected a noninteger or positive integer, got {f:?}"),
97
+ )
98
+ .into())
99
+ } else {
100
+ Ok(())
101
+ }
102
+ }
103
+
104
+ /// Math module functions — each variant corresponds to a Python-visible function.
105
+ #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, strum::Display, serde::Serialize, serde::Deserialize)]
106
+ #[strum(serialize_all = "lowercase")]
107
+ pub(crate) enum MathFunctions {
108
+ // Rounding
109
+ Floor,
110
+ Ceil,
111
+ Trunc,
112
+ // Roots & powers
113
+ Sqrt,
114
+ Isqrt,
115
+ Cbrt,
116
+ Pow,
117
+ Exp,
118
+ Exp2,
119
+ Expm1,
120
+ // Logarithms
121
+ Log,
122
+ Log1p,
123
+ Log2,
124
+ Log10,
125
+ // Float properties
126
+ Fabs,
127
+ Isnan,
128
+ Isinf,
129
+ Isfinite,
130
+ Copysign,
131
+ Isclose,
132
+ Nextafter,
133
+ Ulp,
134
+ // Trigonometric
135
+ Sin,
136
+ Cos,
137
+ Tan,
138
+ Asin,
139
+ Acos,
140
+ Atan,
141
+ Atan2,
142
+ // Hyperbolic
143
+ Sinh,
144
+ Cosh,
145
+ Tanh,
146
+ Asinh,
147
+ Acosh,
148
+ Atanh,
149
+ // Angular conversion
150
+ Degrees,
151
+ Radians,
152
+ // Integer math
153
+ Factorial,
154
+ Gcd,
155
+ Lcm,
156
+ Comb,
157
+ Perm,
158
+ // Modular / decomposition
159
+ Fmod,
160
+ Remainder,
161
+ Modf,
162
+ Frexp,
163
+ Ldexp,
164
+ // Special functions
165
+ Gamma,
166
+ Lgamma,
167
+ Erf,
168
+ Erfc,
169
+ }
170
+
171
+ /// Creates the `math` module and allocates it on the heap.
172
+ ///
173
+ /// Registers all math functions and constants (`pi`, `e`, `tau`, `inf`, `nan`)
174
+ /// matching CPython's `math` module. Functions are registered as
175
+ /// `ModuleFunctions::Math` variants.
176
+ ///
177
+ /// # Returns
178
+ /// A `HeapId` pointing to the newly allocated module.
179
+ ///
180
+ /// # Panics
181
+ /// Panics if the required strings have not been pre-interned during prepare phase.
182
+ pub fn create_module(vm: &mut VM<'_, '_, impl ResourceTracker>) -> Result<HeapId, ResourceError> {
183
+ let mut module = Module::new(StaticStrings::Math);
184
+
185
+ // Register all math functions
186
+ for (name, func) in MATH_FUNCTIONS {
187
+ module.set_attr(*name, Value::ModuleFunction(ModuleFunctions::Math(*func)), vm);
188
+ }
189
+
190
+ // Constants
191
+ module.set_attr(StaticStrings::Pi, Value::Float(std::f64::consts::PI), vm);
192
+ module.set_attr(StaticStrings::MathE, Value::Float(std::f64::consts::E), vm);
193
+ module.set_attr(StaticStrings::Tau, Value::Float(std::f64::consts::TAU), vm);
194
+ module.set_attr(StaticStrings::MathInf, Value::Float(f64::INFINITY), vm);
195
+ module.set_attr(StaticStrings::MathNan, Value::Float(f64::NAN), vm);
196
+
197
+ vm.heap.allocate(HeapData::Module(module))
198
+ }
199
+
200
+ /// Static mapping of attribute names to math functions for module creation.
201
+ const MATH_FUNCTIONS: &[(StaticStrings, MathFunctions)] = &[
202
+ // Rounding
203
+ (StaticStrings::Floor, MathFunctions::Floor),
204
+ (StaticStrings::Ceil, MathFunctions::Ceil),
205
+ (StaticStrings::Trunc, MathFunctions::Trunc),
206
+ // Roots & powers
207
+ (StaticStrings::Sqrt, MathFunctions::Sqrt),
208
+ (StaticStrings::Isqrt, MathFunctions::Isqrt),
209
+ (StaticStrings::Cbrt, MathFunctions::Cbrt),
210
+ (StaticStrings::Pow, MathFunctions::Pow),
211
+ (StaticStrings::Exp, MathFunctions::Exp),
212
+ (StaticStrings::Exp2, MathFunctions::Exp2),
213
+ (StaticStrings::Expm1, MathFunctions::Expm1),
214
+ // Logarithms
215
+ (StaticStrings::Log, MathFunctions::Log),
216
+ (StaticStrings::Log1p, MathFunctions::Log1p),
217
+ (StaticStrings::Log2, MathFunctions::Log2),
218
+ (StaticStrings::Log10, MathFunctions::Log10),
219
+ // Float properties
220
+ (StaticStrings::Fabs, MathFunctions::Fabs),
221
+ (StaticStrings::Isnan, MathFunctions::Isnan),
222
+ (StaticStrings::Isinf, MathFunctions::Isinf),
223
+ (StaticStrings::Isfinite, MathFunctions::Isfinite),
224
+ (StaticStrings::Copysign, MathFunctions::Copysign),
225
+ (StaticStrings::Isclose, MathFunctions::Isclose),
226
+ (StaticStrings::Nextafter, MathFunctions::Nextafter),
227
+ (StaticStrings::Ulp, MathFunctions::Ulp),
228
+ // Trigonometric
229
+ (StaticStrings::Sin, MathFunctions::Sin),
230
+ (StaticStrings::Cos, MathFunctions::Cos),
231
+ (StaticStrings::Tan, MathFunctions::Tan),
232
+ (StaticStrings::Asin, MathFunctions::Asin),
233
+ (StaticStrings::Acos, MathFunctions::Acos),
234
+ (StaticStrings::Atan, MathFunctions::Atan),
235
+ (StaticStrings::Atan2, MathFunctions::Atan2),
236
+ // Hyperbolic
237
+ (StaticStrings::Sinh, MathFunctions::Sinh),
238
+ (StaticStrings::Cosh, MathFunctions::Cosh),
239
+ (StaticStrings::Tanh, MathFunctions::Tanh),
240
+ (StaticStrings::Asinh, MathFunctions::Asinh),
241
+ (StaticStrings::Acosh, MathFunctions::Acosh),
242
+ (StaticStrings::Atanh, MathFunctions::Atanh),
243
+ // Angular conversion
244
+ (StaticStrings::Degrees, MathFunctions::Degrees),
245
+ (StaticStrings::Radians, MathFunctions::Radians),
246
+ // Integer math
247
+ (StaticStrings::Factorial, MathFunctions::Factorial),
248
+ (StaticStrings::Gcd, MathFunctions::Gcd),
249
+ (StaticStrings::Lcm, MathFunctions::Lcm),
250
+ (StaticStrings::Comb, MathFunctions::Comb),
251
+ (StaticStrings::Perm, MathFunctions::Perm),
252
+ // Modular / decomposition
253
+ (StaticStrings::Fmod, MathFunctions::Fmod),
254
+ (StaticStrings::Remainder, MathFunctions::Remainder),
255
+ (StaticStrings::Modf, MathFunctions::Modf),
256
+ (StaticStrings::Frexp, MathFunctions::Frexp),
257
+ (StaticStrings::Ldexp, MathFunctions::Ldexp),
258
+ // Special functions
259
+ (StaticStrings::Gamma, MathFunctions::Gamma),
260
+ (StaticStrings::Lgamma, MathFunctions::Lgamma),
261
+ (StaticStrings::Erf, MathFunctions::Erf),
262
+ (StaticStrings::Erfc, MathFunctions::Erfc),
263
+ ];
264
+
265
+ /// Dispatches a call to a math module function.
266
+ ///
267
+ /// All math functions are pure computations and return `Value` directly.
268
+ pub(super) fn call(
269
+ vm: &mut VM<'_, '_, impl ResourceTracker>,
270
+ function: MathFunctions,
271
+ args: ArgValues,
272
+ ) -> RunResult<Value> {
273
+ match function {
274
+ // Rounding
275
+ MathFunctions::Floor => math_floor(vm.heap, args),
276
+ MathFunctions::Ceil => math_ceil(vm.heap, args),
277
+ MathFunctions::Trunc => math_trunc(vm.heap, args),
278
+ // Roots & powers
279
+ MathFunctions::Sqrt => math_sqrt(vm.heap, args),
280
+ MathFunctions::Isqrt => math_isqrt(vm.heap, args),
281
+ MathFunctions::Cbrt => math_cbrt(vm.heap, args),
282
+ MathFunctions::Pow => math_pow(vm.heap, args),
283
+ MathFunctions::Exp => math_exp(vm.heap, args),
284
+ MathFunctions::Exp2 => math_exp2(vm.heap, args),
285
+ MathFunctions::Expm1 => math_expm1(vm.heap, args),
286
+ // Logarithms
287
+ MathFunctions::Log => math_log(vm.heap, args),
288
+ MathFunctions::Log1p => math_log1p(vm.heap, args),
289
+ MathFunctions::Log2 => math_log2(vm.heap, args),
290
+ MathFunctions::Log10 => math_log10(vm.heap, args),
291
+ // Float properties
292
+ MathFunctions::Fabs => math_fabs(vm.heap, args),
293
+ MathFunctions::Isnan => math_isnan(vm.heap, args),
294
+ MathFunctions::Isinf => math_isinf(vm.heap, args),
295
+ MathFunctions::Isfinite => math_isfinite(vm.heap, args),
296
+ MathFunctions::Copysign => math_copysign(vm.heap, args),
297
+ MathFunctions::Isclose => math_isclose(vm.heap, args, vm.interns),
298
+ MathFunctions::Nextafter => math_nextafter(vm.heap, args),
299
+ MathFunctions::Ulp => math_ulp(vm.heap, args),
300
+ // Trigonometric
301
+ MathFunctions::Sin => math_sin(vm.heap, args),
302
+ MathFunctions::Cos => math_cos(vm.heap, args),
303
+ MathFunctions::Tan => math_tan(vm.heap, args),
304
+ MathFunctions::Asin => math_asin(vm.heap, args),
305
+ MathFunctions::Acos => math_acos(vm.heap, args),
306
+ MathFunctions::Atan => math_atan(vm.heap, args),
307
+ MathFunctions::Atan2 => math_atan2(vm.heap, args),
308
+ // Hyperbolic
309
+ MathFunctions::Sinh => math_sinh(vm.heap, args),
310
+ MathFunctions::Cosh => math_cosh(vm.heap, args),
311
+ MathFunctions::Tanh => math_tanh(vm.heap, args),
312
+ MathFunctions::Asinh => math_asinh(vm.heap, args),
313
+ MathFunctions::Acosh => math_acosh(vm.heap, args),
314
+ MathFunctions::Atanh => math_atanh(vm.heap, args),
315
+ // Angular conversion
316
+ MathFunctions::Degrees => math_degrees(vm.heap, args),
317
+ MathFunctions::Radians => math_radians(vm.heap, args),
318
+ // Integer math
319
+ MathFunctions::Factorial => math_factorial(vm.heap, args),
320
+ MathFunctions::Gcd => math_gcd(vm.heap, args),
321
+ MathFunctions::Lcm => math_lcm(vm.heap, args),
322
+ MathFunctions::Comb => math_comb(vm.heap, args),
323
+ MathFunctions::Perm => math_perm(vm.heap, args),
324
+ // Modular / decomposition
325
+ MathFunctions::Fmod => math_fmod(vm.heap, args),
326
+ MathFunctions::Remainder => math_remainder(vm.heap, args),
327
+ MathFunctions::Modf => math_modf(vm.heap, args),
328
+ MathFunctions::Frexp => math_frexp(vm.heap, args),
329
+ MathFunctions::Ldexp => math_ldexp(vm.heap, args),
330
+ // Special functions
331
+ MathFunctions::Gamma => math_gamma(vm.heap, args),
332
+ MathFunctions::Lgamma => math_lgamma(vm.heap, args),
333
+ MathFunctions::Erf => math_erf(vm.heap, args),
334
+ MathFunctions::Erfc => math_erfc(vm.heap, args),
335
+ }
336
+ }
337
+
338
+ // ==========================
339
+ // Rounding functions
340
+ // ==========================
341
+
342
+ /// `math.floor(x)` — returns the largest integer less than or equal to x.
343
+ ///
344
+ /// Accepts int, float, or bool. Returns int.
345
+ /// Raises `OverflowError` for infinity, `ValueError` for NaN.
346
+ fn math_floor(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
347
+ let value = args.get_one_arg("math.floor", heap)?;
348
+ defer_drop!(value, heap);
349
+
350
+ match value {
351
+ Value::Float(f) => float_to_int_checked(f.floor(), *f, heap),
352
+ Value::Int(n) => Ok(Value::Int(*n)),
353
+ Value::Bool(b) => Ok(Value::Int(i64::from(*b))),
354
+ _ => Err(ExcType::type_error(format!(
355
+ "must be real number, not {}",
356
+ value.py_type(heap)
357
+ ))),
358
+ }
359
+ }
360
+
361
+ /// `math.ceil(x)` — returns the smallest integer greater than or equal to x.
362
+ ///
363
+ /// Accepts int, float, or bool. Returns int.
364
+ /// Raises `OverflowError` for infinity, `ValueError` for NaN.
365
+ fn math_ceil(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
366
+ let value = args.get_one_arg("math.ceil", heap)?;
367
+ defer_drop!(value, heap);
368
+
369
+ match value {
370
+ Value::Float(f) => float_to_int_checked(f.ceil(), *f, heap),
371
+ Value::Int(n) => Ok(Value::Int(*n)),
372
+ Value::Bool(b) => Ok(Value::Int(i64::from(*b))),
373
+ _ => Err(ExcType::type_error(format!(
374
+ "must be real number, not {}",
375
+ value.py_type(heap)
376
+ ))),
377
+ }
378
+ }
379
+
380
+ /// `math.trunc(x)` — truncates x to the nearest integer toward zero.
381
+ ///
382
+ /// Accepts int, float, or bool. Returns int.
383
+ fn math_trunc(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
384
+ let value = args.get_one_arg("math.trunc", heap)?;
385
+ defer_drop!(value, heap);
386
+
387
+ match value {
388
+ Value::Float(f) => float_to_int_checked(f.trunc(), *f, heap),
389
+ Value::Int(n) => Ok(Value::Int(*n)),
390
+ Value::Bool(b) => Ok(Value::Int(i64::from(*b))),
391
+ _ => Err(ExcType::type_error(format!(
392
+ "type {} doesn't define __trunc__ method",
393
+ value.py_type(heap)
394
+ ))),
395
+ }
396
+ }
397
+
398
+ // ==========================
399
+ // Roots & powers
400
+ // ==========================
401
+
402
+ /// `math.sqrt(x)` — returns the square root of x.
403
+ ///
404
+ /// Always returns a float. Raises `ValueError` for negative inputs with a
405
+ /// descriptive message matching CPython 3.14: "expected a nonnegative input, got <x>".
406
+ fn math_sqrt(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
407
+ let value = args.get_one_arg("math.sqrt", heap)?;
408
+ defer_drop!(value, heap);
409
+
410
+ let f = value_to_float(value, heap)?;
411
+ if f < 0.0 {
412
+ Err(SimpleException::new_msg(ExcType::ValueError, format!("expected a nonnegative input, got {f:?}")).into())
413
+ } else {
414
+ Ok(Value::Float(f.sqrt()))
415
+ }
416
+ }
417
+
418
+ /// `math.isqrt(n)` — returns the integer square root of a non-negative integer.
419
+ ///
420
+ /// Returns the largest integer `r` such that `r * r <= n`.
421
+ /// Only accepts non-negative integers (and bools).
422
+ fn math_isqrt(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
423
+ let value = args.get_one_arg("math.isqrt", heap)?;
424
+ defer_drop!(value, heap);
425
+
426
+ let n = value_to_int(value, heap)?;
427
+ if n < 0 {
428
+ return Err(SimpleException::new_msg(ExcType::ValueError, "isqrt() argument must be nonnegative").into());
429
+ }
430
+ if n == 0 {
431
+ return Ok(Value::Int(0));
432
+ }
433
+
434
+ // Integer square root via f64 estimate + correction.
435
+ // For i64 inputs, f64 sqrt is accurate to within ±1, so we need to
436
+ // correct both overshoot and undershoot. The cast truncates toward zero,
437
+ // so undershoot is possible for perfect squares near f64 precision limits.
438
+ #[expect(
439
+ clippy::cast_precision_loss,
440
+ clippy::cast_possible_truncation,
441
+ reason = "initial estimate doesn't need to be exact, correction refines it"
442
+ )]
443
+ let mut x = (n as f64).sqrt() as i64;
444
+ // Correct overshoot: use `x > n / x` instead of `x * x > n` to avoid i64 overflow.
445
+ while x > n / x {
446
+ x -= 1;
447
+ }
448
+ // Correct undershoot: check if (x+1)² ≤ n using division to avoid overflow.
449
+ while x < n / (x + 1) {
450
+ x += 1;
451
+ }
452
+ Ok(Value::Int(x))
453
+ }
454
+
455
+ /// `math.cbrt(x)` — returns the cube root of x.
456
+ ///
457
+ /// Always returns a float. Unlike `sqrt`, works for negative inputs.
458
+ fn math_cbrt(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
459
+ let value = args.get_one_arg("math.cbrt", heap)?;
460
+ defer_drop!(value, heap);
461
+
462
+ let f = value_to_float(value, heap)?;
463
+ Ok(Value::Float(f.cbrt()))
464
+ }
465
+
466
+ /// `math.pow(x, y)` — returns x raised to the power y.
467
+ ///
468
+ /// Always returns a float. Unlike the builtin `pow()`, does not support
469
+ /// three-argument modular exponentiation. Raises `ValueError` for
470
+ /// negative base with non-integer exponent.
471
+ fn math_pow(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
472
+ let (x_val, y_val) = args.get_two_args("math.pow", heap)?;
473
+ defer_drop!(x_val, heap);
474
+ defer_drop!(y_val, heap);
475
+
476
+ let x = value_to_float(x_val, heap)?;
477
+ let y = value_to_float(y_val, heap)?;
478
+ let result = x.powf(y);
479
+ // CPython raises ValueError for domain errors: 0**negative, negative**non-integer
480
+ if result.is_nan() && !x.is_nan() && !y.is_nan() {
481
+ return Err(math_domain_error());
482
+ }
483
+ if result.is_infinite() && x.is_finite() && y.is_finite() {
484
+ // 0**negative is a domain error (ValueError), not overflow
485
+ if x == 0.0 && y < 0.0 {
486
+ return Err(math_domain_error());
487
+ }
488
+ return Err(math_range_error());
489
+ }
490
+ Ok(Value::Float(result))
491
+ }
492
+
493
+ /// `math.exp(x)` — returns e raised to the power x.
494
+ fn math_exp(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
495
+ let value = args.get_one_arg("math.exp", heap)?;
496
+ defer_drop!(value, heap);
497
+
498
+ let f = value_to_float(value, heap)?;
499
+ let result = f.exp();
500
+ check_range_error(result, f)?;
501
+ Ok(Value::Float(result))
502
+ }
503
+
504
+ /// `math.exp2(x)` — returns 2 raised to the power x.
505
+ fn math_exp2(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
506
+ let value = args.get_one_arg("math.exp2", heap)?;
507
+ defer_drop!(value, heap);
508
+
509
+ let f = value_to_float(value, heap)?;
510
+ let result = f.exp2();
511
+ check_range_error(result, f)?;
512
+ Ok(Value::Float(result))
513
+ }
514
+
515
+ /// `math.expm1(x)` — returns e**x - 1.
516
+ ///
517
+ /// More accurate than `exp(x) - 1` for small values of x.
518
+ fn math_expm1(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
519
+ let value = args.get_one_arg("math.expm1", heap)?;
520
+ defer_drop!(value, heap);
521
+
522
+ let f = value_to_float(value, heap)?;
523
+ let result = f.exp_m1();
524
+ check_range_error(result, f)?;
525
+ Ok(Value::Float(result))
526
+ }
527
+
528
+ // ==========================
529
+ // Logarithms
530
+ // ==========================
531
+
532
+ /// `math.log(x[, base])` — returns the logarithm of x.
533
+ ///
534
+ /// With one argument, returns the natural logarithm (base e).
535
+ /// With two arguments, returns `log(x) / log(base)`.
536
+ /// Raises `ValueError` for non-positive inputs (CPython 3.14: "expected a positive input").
537
+ fn math_log(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
538
+ let (x_val, base_val) = args.get_one_two_args("math.log", heap)?;
539
+ defer_drop!(x_val, heap);
540
+ defer_drop!(base_val, heap);
541
+
542
+ let x = value_to_float(x_val, heap)?;
543
+ if x <= 0.0 {
544
+ return Err(SimpleException::new_msg(ExcType::ValueError, "expected a positive input").into());
545
+ }
546
+
547
+ match base_val {
548
+ Some(base_v) => {
549
+ let base = value_to_float(base_v, heap)?;
550
+ // base == 1.0 causes division by zero in log(x)/log(base), matching
551
+ // CPython which raises ZeroDivisionError for this case.
552
+ #[expect(
553
+ clippy::float_cmp,
554
+ reason = "exact comparison with 1.0 is intentional — log(1.0) is exactly 0.0"
555
+ )]
556
+ if base == 1.0 {
557
+ return Err(SimpleException::new_msg(ExcType::ZeroDivisionError, "division by zero").into());
558
+ }
559
+ if base <= 0.0 {
560
+ return Err(SimpleException::new_msg(ExcType::ValueError, "expected a positive input").into());
561
+ }
562
+ Ok(Value::Float(x.ln() / base.ln()))
563
+ }
564
+ None => Ok(Value::Float(x.ln())),
565
+ }
566
+ }
567
+
568
+ /// `math.log1p(x)` — returns the natural logarithm of 1 + x.
569
+ ///
570
+ /// More accurate than `log(1 + x)` for small values of x.
571
+ /// CPython 3.14 raises ValueError with "expected argument value > -1, got <x>".
572
+ fn math_log1p(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
573
+ let value = args.get_one_arg("math.log1p", heap)?;
574
+ defer_drop!(value, heap);
575
+
576
+ let f = value_to_float(value, heap)?;
577
+ if f <= -1.0 {
578
+ return Err(
579
+ SimpleException::new_msg(ExcType::ValueError, format!("expected argument value > -1, got {f:?}")).into(),
580
+ );
581
+ }
582
+ Ok(Value::Float(f.ln_1p()))
583
+ }
584
+
585
+ /// `math.log2(x)` — returns the base-2 logarithm of x.
586
+ ///
587
+ /// Returns `inf` for positive infinity, `nan` for NaN.
588
+ /// Raises `ValueError` for non-positive finite inputs.
589
+ fn math_log2(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
590
+ let value = args.get_one_arg("math.log2", heap)?;
591
+ defer_drop!(value, heap);
592
+
593
+ let f = value_to_float(value, heap)?;
594
+ if f <= 0.0 {
595
+ Err(SimpleException::new_msg(ExcType::ValueError, "expected a positive input").into())
596
+ } else {
597
+ Ok(Value::Float(f.log2()))
598
+ }
599
+ }
600
+
601
+ /// `math.log10(x)` — returns the base-10 logarithm of x.
602
+ ///
603
+ /// Returns `inf` for positive infinity, `nan` for NaN.
604
+ /// Raises `ValueError` for non-positive finite inputs.
605
+ fn math_log10(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
606
+ let value = args.get_one_arg("math.log10", heap)?;
607
+ defer_drop!(value, heap);
608
+
609
+ let f = value_to_float(value, heap)?;
610
+ if f <= 0.0 {
611
+ Err(SimpleException::new_msg(ExcType::ValueError, "expected a positive input").into())
612
+ } else {
613
+ Ok(Value::Float(f.log10()))
614
+ }
615
+ }
616
+
617
+ // ==========================
618
+ // Float properties
619
+ // ==========================
620
+
621
+ /// `math.fabs(x)` — returns the absolute value as a float.
622
+ ///
623
+ /// Unlike the builtin `abs()`, always returns a float.
624
+ fn math_fabs(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
625
+ let value = args.get_one_arg("math.fabs", heap)?;
626
+ defer_drop!(value, heap);
627
+
628
+ let f = value_to_float(value, heap)?;
629
+ Ok(Value::Float(f.abs()))
630
+ }
631
+
632
+ /// `math.isnan(x)` — returns True if x is NaN.
633
+ fn math_isnan(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
634
+ let value = args.get_one_arg("math.isnan", heap)?;
635
+ defer_drop!(value, heap);
636
+
637
+ let f = value_to_float(value, heap)?;
638
+ Ok(Value::Bool(f.is_nan()))
639
+ }
640
+
641
+ /// `math.isinf(x)` — returns True if x is positive or negative infinity.
642
+ fn math_isinf(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
643
+ let value = args.get_one_arg("math.isinf", heap)?;
644
+ defer_drop!(value, heap);
645
+
646
+ let f = value_to_float(value, heap)?;
647
+ Ok(Value::Bool(f.is_infinite()))
648
+ }
649
+
650
+ /// `math.isfinite(x)` — returns True if x is neither infinity nor NaN.
651
+ fn math_isfinite(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
652
+ let value = args.get_one_arg("math.isfinite", heap)?;
653
+ defer_drop!(value, heap);
654
+
655
+ let f = value_to_float(value, heap)?;
656
+ Ok(Value::Bool(f.is_finite()))
657
+ }
658
+
659
+ /// `math.copysign(x, y)` — returns x with the sign of y.
660
+ ///
661
+ /// Always returns a float.
662
+ fn math_copysign(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
663
+ let (x_val, y_val) = args.get_two_args("math.copysign", heap)?;
664
+ defer_drop!(x_val, heap);
665
+ defer_drop!(y_val, heap);
666
+
667
+ let x = value_to_float(x_val, heap)?;
668
+ let y = value_to_float(y_val, heap)?;
669
+ Ok(Value::Float(x.copysign(y)))
670
+ }
671
+
672
+ /// `math.isclose(a, b, *, rel_tol=1e-9, abs_tol=0.0)` — returns True if a and b are close.
673
+ ///
674
+ /// Supports keyword-only `rel_tol` and `abs_tol` parameters matching CPython.
675
+ /// Raises `ValueError` if either tolerance is negative.
676
+ fn math_isclose(heap: &mut Heap<impl ResourceTracker>, args: ArgValues, interns: &Interns) -> RunResult<Value> {
677
+ let (positional, kwargs) = args.into_parts();
678
+ defer_drop_mut!(positional, heap);
679
+
680
+ // Extract exactly two positional args
681
+ let Some(a_val) = positional.next() else {
682
+ return Err(ExcType::type_error_at_least("math.isclose", 2, 0));
683
+ };
684
+ defer_drop!(a_val, heap);
685
+ let Some(b_val) = positional.next() else {
686
+ return Err(ExcType::type_error_at_least("math.isclose", 2, 1));
687
+ };
688
+ defer_drop!(b_val, heap);
689
+ if positional.len() > 0 {
690
+ return Err(ExcType::type_error_at_most("math.isclose", 2, 2 + positional.len()));
691
+ }
692
+
693
+ let a = value_to_float(a_val, heap)?;
694
+ let b = value_to_float(b_val, heap)?;
695
+
696
+ // Parse optional keyword arguments rel_tol and abs_tol
697
+ let (rel_tol, abs_tol) = extract_isclose_kwargs(kwargs, heap, interns)?;
698
+
699
+ if rel_tol < 0.0 {
700
+ return Err(SimpleException::new_msg(ExcType::ValueError, "tolerances must be non-negative").into());
701
+ }
702
+ if abs_tol < 0.0 {
703
+ return Err(SimpleException::new_msg(ExcType::ValueError, "tolerances must be non-negative").into());
704
+ }
705
+
706
+ // Exact equality check matches CPython's isclose() behavior — two identical
707
+ // values (including infinities) are always considered close.
708
+ #[expect(
709
+ clippy::float_cmp,
710
+ reason = "exact equality check matches CPython's isclose() semantics"
711
+ )]
712
+ if a == b {
713
+ return Ok(Value::Bool(true));
714
+ }
715
+ if a.is_infinite() || b.is_infinite() {
716
+ return Ok(Value::Bool(false));
717
+ }
718
+ if a.is_nan() || b.is_nan() {
719
+ return Ok(Value::Bool(false));
720
+ }
721
+
722
+ let diff = (a - b).abs();
723
+ let result = diff <= (rel_tol * a.abs().max(b.abs())).max(abs_tol);
724
+ Ok(Value::Bool(result))
725
+ }
726
+
727
+ /// Extracts `rel_tol` and `abs_tol` keyword arguments for `math.isclose`.
728
+ ///
729
+ /// Returns `(rel_tol, abs_tol)` with defaults of `(1e-9, 0.0)`.
730
+ fn extract_isclose_kwargs(
731
+ kwargs: crate::args::KwargsValues,
732
+ heap: &mut Heap<impl ResourceTracker>,
733
+ interns: &Interns,
734
+ ) -> RunResult<(f64, f64)> {
735
+ let mut rel_tol: f64 = 1e-9;
736
+ let mut abs_tol: f64 = 0.0;
737
+
738
+ if kwargs.is_empty() {
739
+ return Ok((rel_tol, abs_tol));
740
+ }
741
+
742
+ for (key, value) in kwargs {
743
+ defer_drop!(key, heap);
744
+ defer_drop!(value, heap);
745
+
746
+ let Some(keyword_name) = key.as_either_str(heap) else {
747
+ return Err(ExcType::type_error("keywords must be strings"));
748
+ };
749
+
750
+ let key_str = keyword_name.as_str(interns);
751
+ match key_str {
752
+ "rel_tol" => {
753
+ rel_tol = value_to_float(value, heap)?;
754
+ }
755
+ "abs_tol" => {
756
+ abs_tol = value_to_float(value, heap)?;
757
+ }
758
+ other => {
759
+ return Err(ExcType::type_error(format!(
760
+ "isclose() got an unexpected keyword argument '{other}'"
761
+ )));
762
+ }
763
+ }
764
+ }
765
+
766
+ Ok((rel_tol, abs_tol))
767
+ }
768
+
769
+ /// `math.nextafter(x, y)` — returns the next float after x towards y.
770
+ fn math_nextafter(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
771
+ let (x_val, y_val) = args.get_two_args("math.nextafter", heap)?;
772
+ defer_drop!(x_val, heap);
773
+ defer_drop!(y_val, heap);
774
+
775
+ let x = value_to_float(x_val, heap)?;
776
+ let y = value_to_float(y_val, heap)?;
777
+
778
+ Ok(Value::Float(libm::nextafter(x, y)))
779
+ }
780
+
781
+ /// `math.ulp(x)` — returns the value of the least significant bit of x.
782
+ ///
783
+ /// For finite non-zero x, returns the smallest float `u` such that `x + u != x`.
784
+ /// Special cases: `ulp(nan)` returns nan, `ulp(inf)` returns inf.
785
+ fn math_ulp(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
786
+ let value = args.get_one_arg("math.ulp", heap)?;
787
+ defer_drop!(value, heap);
788
+
789
+ let f = value_to_float(value, heap)?;
790
+ if f.is_nan() {
791
+ return Ok(Value::Float(f64::NAN));
792
+ }
793
+ if f.is_infinite() {
794
+ return Ok(Value::Float(f64::INFINITY));
795
+ }
796
+ let f = f.abs();
797
+ if f == 0.0 {
798
+ // CPython returns the smallest positive subnormal: 5e-324
799
+ return Ok(Value::Float(f64::from_bits(1)));
800
+ }
801
+ // ULP = nextafter(f, inf) - f
802
+ let next = libm::nextafter(f, f64::INFINITY);
803
+ Ok(Value::Float(next - f))
804
+ }
805
+
806
+ // ==========================
807
+ // Trigonometric functions
808
+ // ==========================
809
+
810
+ /// `math.sin(x)` — returns the sine of x (in radians).
811
+ ///
812
+ /// CPython 3.14 raises ValueError for infinity: "expected a finite input, got inf".
813
+ fn math_sin(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
814
+ let value = args.get_one_arg("math.sin", heap)?;
815
+ defer_drop!(value, heap);
816
+
817
+ let f = value_to_float(value, heap)?;
818
+ require_finite(f)?;
819
+ Ok(Value::Float(f.sin()))
820
+ }
821
+
822
+ /// `math.cos(x)` — returns the cosine of x (in radians).
823
+ fn math_cos(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
824
+ let value = args.get_one_arg("math.cos", heap)?;
825
+ defer_drop!(value, heap);
826
+
827
+ let f = value_to_float(value, heap)?;
828
+ require_finite(f)?;
829
+ Ok(Value::Float(f.cos()))
830
+ }
831
+
832
+ /// `math.tan(x)` — returns the tangent of x (in radians).
833
+ fn math_tan(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
834
+ let value = args.get_one_arg("math.tan", heap)?;
835
+ defer_drop!(value, heap);
836
+
837
+ let f = value_to_float(value, heap)?;
838
+ require_finite(f)?;
839
+ Ok(Value::Float(f.tan()))
840
+ }
841
+
842
+ /// `math.asin(x)` — returns the arc sine of x (in radians).
843
+ ///
844
+ /// CPython 3.14: "expected a number in range from -1 up to 1, got <x>".
845
+ fn math_asin(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
846
+ let value = args.get_one_arg("math.asin", heap)?;
847
+ defer_drop!(value, heap);
848
+
849
+ let f = value_to_float(value, heap)?;
850
+ require_unit_range(f)?;
851
+ Ok(Value::Float(f.asin()))
852
+ }
853
+
854
+ /// `math.acos(x)` — returns the arc cosine of x (in radians).
855
+ ///
856
+ /// CPython 3.14: "expected a number in range from -1 up to 1, got <x>".
857
+ fn math_acos(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
858
+ let value = args.get_one_arg("math.acos", heap)?;
859
+ defer_drop!(value, heap);
860
+
861
+ let f = value_to_float(value, heap)?;
862
+ require_unit_range(f)?;
863
+ Ok(Value::Float(f.acos()))
864
+ }
865
+
866
+ /// `math.atan(x)` — returns the arc tangent of x (in radians).
867
+ fn math_atan(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
868
+ let value = args.get_one_arg("math.atan", heap)?;
869
+ defer_drop!(value, heap);
870
+
871
+ let f = value_to_float(value, heap)?;
872
+ Ok(Value::Float(f.atan()))
873
+ }
874
+
875
+ /// `math.atan2(y, x)` — returns atan(y/x) in radians, using the signs of both
876
+ /// to determine the correct quadrant.
877
+ fn math_atan2(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
878
+ let (y_val, x_val) = args.get_two_args("math.atan2", heap)?;
879
+ defer_drop!(y_val, heap);
880
+ defer_drop!(x_val, heap);
881
+
882
+ let y = value_to_float(y_val, heap)?;
883
+ let x = value_to_float(x_val, heap)?;
884
+ Ok(Value::Float(y.atan2(x)))
885
+ }
886
+
887
+ // ==========================
888
+ // Hyperbolic functions
889
+ // ==========================
890
+
891
+ /// `math.sinh(x)` — returns the hyperbolic sine of x.
892
+ fn math_sinh(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
893
+ let value = args.get_one_arg("math.sinh", heap)?;
894
+ defer_drop!(value, heap);
895
+
896
+ let f = value_to_float(value, heap)?;
897
+ let result = f.sinh();
898
+ check_range_error(result, f)?;
899
+ Ok(Value::Float(result))
900
+ }
901
+
902
+ /// `math.cosh(x)` — returns the hyperbolic cosine of x.
903
+ fn math_cosh(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
904
+ let value = args.get_one_arg("math.cosh", heap)?;
905
+ defer_drop!(value, heap);
906
+
907
+ let f = value_to_float(value, heap)?;
908
+ let result = f.cosh();
909
+ check_range_error(result, f)?;
910
+ Ok(Value::Float(result))
911
+ }
912
+
913
+ /// `math.tanh(x)` — returns the hyperbolic tangent of x.
914
+ fn math_tanh(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
915
+ let value = args.get_one_arg("math.tanh", heap)?;
916
+ defer_drop!(value, heap);
917
+
918
+ let f = value_to_float(value, heap)?;
919
+ Ok(Value::Float(f.tanh()))
920
+ }
921
+
922
+ /// `math.asinh(x)` — returns the inverse hyperbolic sine of x.
923
+ fn math_asinh(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
924
+ let value = args.get_one_arg("math.asinh", heap)?;
925
+ defer_drop!(value, heap);
926
+
927
+ let f = value_to_float(value, heap)?;
928
+ Ok(Value::Float(f.asinh()))
929
+ }
930
+
931
+ /// `math.acosh(x)` — returns the inverse hyperbolic cosine of x.
932
+ ///
933
+ /// CPython 3.14: "expected argument value not less than 1, got <x>".
934
+ fn math_acosh(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
935
+ let value = args.get_one_arg("math.acosh", heap)?;
936
+ defer_drop!(value, heap);
937
+
938
+ let f = value_to_float(value, heap)?;
939
+ if f < 1.0 {
940
+ return Err(SimpleException::new_msg(
941
+ ExcType::ValueError,
942
+ format!("expected argument value not less than 1, got {f:?}"),
943
+ )
944
+ .into());
945
+ }
946
+ Ok(Value::Float(f.acosh()))
947
+ }
948
+
949
+ /// `math.atanh(x)` — returns the inverse hyperbolic tangent of x.
950
+ ///
951
+ /// CPython 3.14: "expected a number between -1 and 1, got <x>".
952
+ fn math_atanh(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
953
+ let value = args.get_one_arg("math.atanh", heap)?;
954
+ defer_drop!(value, heap);
955
+
956
+ let f = value_to_float(value, heap)?;
957
+ if f <= -1.0 || f >= 1.0 {
958
+ return Err(SimpleException::new_msg(
959
+ ExcType::ValueError,
960
+ format!("expected a number between -1 and 1, got {f:?}"),
961
+ )
962
+ .into());
963
+ }
964
+ Ok(Value::Float(f.atanh()))
965
+ }
966
+
967
+ // ==========================
968
+ // Angular conversion
969
+ // ==========================
970
+
971
+ /// `math.degrees(x)` — converts angle x from radians to degrees.
972
+ fn math_degrees(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
973
+ let value = args.get_one_arg("math.degrees", heap)?;
974
+ defer_drop!(value, heap);
975
+
976
+ let f = value_to_float(value, heap)?;
977
+ Ok(Value::Float(f.to_degrees()))
978
+ }
979
+
980
+ /// `math.radians(x)` — converts angle x from degrees to radians.
981
+ fn math_radians(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
982
+ let value = args.get_one_arg("math.radians", heap)?;
983
+ defer_drop!(value, heap);
984
+
985
+ let f = value_to_float(value, heap)?;
986
+ Ok(Value::Float(f.to_radians()))
987
+ }
988
+
989
+ // ==========================
990
+ // Integer math
991
+ // ==========================
992
+
993
+ /// `math.factorial(n)` — returns n factorial.
994
+ ///
995
+ /// Only accepts non-negative integers (and bools). Raises `ValueError` for
996
+ /// negative values, `TypeError` for non-integer types.
997
+ fn math_factorial(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
998
+ let value = args.get_one_arg("math.factorial", heap)?;
999
+ defer_drop!(value, heap);
1000
+
1001
+ let n = match value {
1002
+ Value::Int(n) => *n,
1003
+ Value::Bool(b) => i64::from(*b),
1004
+ _ => {
1005
+ return Err(ExcType::type_error(format!(
1006
+ "'{}' object cannot be interpreted as an integer",
1007
+ value.py_type(heap)
1008
+ )));
1009
+ }
1010
+ };
1011
+
1012
+ if n < 0 {
1013
+ return Err(
1014
+ SimpleException::new_msg(ExcType::ValueError, "factorial() not defined for negative values").into(),
1015
+ );
1016
+ }
1017
+
1018
+ // Compute factorial iteratively
1019
+ let mut result: i64 = 1;
1020
+ for i in 2..=n {
1021
+ match result.checked_mul(i) {
1022
+ Some(v) => result = v,
1023
+ None => {
1024
+ // Overflow — for simplicity, return an error for very large factorials
1025
+ // since we don't have LongInt factorial support yet
1026
+ return Err(
1027
+ SimpleException::new_msg(ExcType::OverflowError, "int too large to convert to factorial").into(),
1028
+ );
1029
+ }
1030
+ }
1031
+ }
1032
+ Ok(Value::Int(result))
1033
+ }
1034
+
1035
+ /// `math.gcd(*integers)` — returns the greatest common divisor of the arguments.
1036
+ ///
1037
+ /// Supports 0 or more arguments, matching CPython 3.9+. `gcd()` returns 0,
1038
+ /// `gcd(n)` returns `abs(n)`, and for multiple args reduces pairwise.
1039
+ /// The result is always non-negative.
1040
+ fn math_gcd(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
1041
+ let positional = args.into_pos_only("math.gcd", heap)?;
1042
+ defer_drop_mut!(positional, heap);
1043
+
1044
+ let mut result: u64 = 0;
1045
+ for arg in positional.by_ref() {
1046
+ defer_drop!(arg, heap);
1047
+ let n = value_to_int(arg, heap)?;
1048
+ result = gcd(result, n.unsigned_abs());
1049
+ }
1050
+ u64_to_value(result, heap)
1051
+ }
1052
+
1053
+ /// `math.lcm(*integers)` — returns the least common multiple of the arguments.
1054
+ ///
1055
+ /// Supports 0 or more arguments, matching CPython 3.9+. `lcm()` returns 1,
1056
+ /// `lcm(n)` returns `abs(n)`, and for multiple args reduces pairwise.
1057
+ /// The result is always non-negative. Returns 0 if any argument is 0.
1058
+ fn math_lcm(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
1059
+ let positional = args.into_pos_only("math.lcm", heap)?;
1060
+ defer_drop_mut!(positional, heap);
1061
+
1062
+ let mut result: u64 = 1;
1063
+ for arg in positional.by_ref() {
1064
+ defer_drop!(arg, heap);
1065
+ let n = value_to_int(arg, heap)?;
1066
+ let abs_n = n.unsigned_abs();
1067
+ if abs_n == 0 {
1068
+ return Ok(Value::Int(0));
1069
+ }
1070
+ let g = gcd(result, abs_n);
1071
+ // lcm(a, b) = |a| / gcd(a,b) * |b| — dividing first avoids intermediate overflow
1072
+ result = (result / g)
1073
+ .checked_mul(abs_n)
1074
+ .ok_or_else(|| SimpleException::new_msg(ExcType::OverflowError, "integer overflow in lcm"))?;
1075
+ }
1076
+ u64_to_value(result, heap)
1077
+ }
1078
+
1079
+ /// `math.comb(n, k)` — returns the number of ways to choose k items from n.
1080
+ ///
1081
+ /// Both arguments must be non-negative integers.
1082
+ fn math_comb(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
1083
+ let (n_val, k_val) = args.get_two_args("math.comb", heap)?;
1084
+ defer_drop!(n_val, heap);
1085
+ defer_drop!(k_val, heap);
1086
+
1087
+ let n = value_to_int(n_val, heap)?;
1088
+ let k = value_to_int(k_val, heap)?;
1089
+
1090
+ if n < 0 {
1091
+ return Err(SimpleException::new_msg(ExcType::ValueError, "n must be a non-negative integer").into());
1092
+ }
1093
+ if k < 0 {
1094
+ return Err(SimpleException::new_msg(ExcType::ValueError, "k must be a non-negative integer").into());
1095
+ }
1096
+ if k > n {
1097
+ return Ok(Value::Int(0));
1098
+ }
1099
+
1100
+ // Use the smaller of k and n-k for efficiency: C(n, k) = C(n, n-k)
1101
+ let k = k.min(n - k);
1102
+ let mut result: i64 = 1;
1103
+ for i in 0..k {
1104
+ // Use GCD reduction to keep intermediates small:
1105
+ // result = result * (n - i) / (i + 1)
1106
+ // By dividing both numerator and denominator by their GCD first,
1107
+ // we reduce the chance of overflow in the multiplication step.
1108
+ let mut numerator = n - i;
1109
+ let mut denominator = i + 1;
1110
+ #[expect(clippy::cast_sign_loss, reason = "both values are known non-negative at this point")]
1111
+ let g = gcd(numerator as u64, denominator as u64).cast_signed();
1112
+ numerator /= g;
1113
+ denominator /= g;
1114
+ // Also reduce against the running result
1115
+ #[expect(clippy::cast_sign_loss, reason = "result and denominator are known non-negative")]
1116
+ let g2 = gcd(result as u64, denominator as u64).cast_signed();
1117
+ result /= g2;
1118
+ denominator /= g2;
1119
+ debug_assert!(denominator == 1, "denominator should be 1 after GCD reduction in comb");
1120
+ match result.checked_mul(numerator) {
1121
+ Some(v) => result = v,
1122
+ None => {
1123
+ return Err(SimpleException::new_msg(ExcType::OverflowError, "integer overflow in comb").into());
1124
+ }
1125
+ }
1126
+ }
1127
+ Ok(Value::Int(result))
1128
+ }
1129
+
1130
+ /// `math.perm(n, k=None)` — returns the number of k-length permutations from n items.
1131
+ ///
1132
+ /// Both arguments must be non-negative integers. When `k` is omitted, defaults to `n`
1133
+ /// (i.e., `perm(n)` returns `n!`), matching CPython behavior.
1134
+ fn math_perm(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
1135
+ let (n_val, k_val) = args.get_one_two_args("math.perm", heap)?;
1136
+ defer_drop!(n_val, heap);
1137
+
1138
+ let n = value_to_int(n_val, heap)?;
1139
+ let k_explicit = k_val.is_some();
1140
+ let k = match k_val {
1141
+ Some(kv) => {
1142
+ defer_drop!(kv, heap);
1143
+ value_to_int(kv, heap)?
1144
+ }
1145
+ None => n,
1146
+ };
1147
+
1148
+ if n < 0 {
1149
+ // When called as perm(n) without k, CPython uses the factorial error message
1150
+ let msg = if k_explicit {
1151
+ "n must be a non-negative integer"
1152
+ } else {
1153
+ "factorial() not defined for negative values"
1154
+ };
1155
+ return Err(SimpleException::new_msg(ExcType::ValueError, msg).into());
1156
+ }
1157
+ if k < 0 {
1158
+ return Err(SimpleException::new_msg(ExcType::ValueError, "k must be a non-negative integer").into());
1159
+ }
1160
+ if k > n {
1161
+ return Ok(Value::Int(0));
1162
+ }
1163
+
1164
+ let mut result: i64 = 1;
1165
+ for i in 0..k {
1166
+ match result.checked_mul(n - i) {
1167
+ Some(v) => result = v,
1168
+ None => {
1169
+ return Err(SimpleException::new_msg(ExcType::OverflowError, "integer overflow in perm").into());
1170
+ }
1171
+ }
1172
+ }
1173
+ Ok(Value::Int(result))
1174
+ }
1175
+
1176
+ // ==========================
1177
+ // Modular / decomposition
1178
+ // ==========================
1179
+
1180
+ /// `math.fmod(x, y)` — returns x modulo y as a float.
1181
+ ///
1182
+ /// Unlike `x % y`, the result has the same sign as x. Raises `ValueError`
1183
+ /// when y is zero (CPython: "math domain error").
1184
+ fn math_fmod(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
1185
+ let (x_val, y_val) = args.get_two_args("math.fmod", heap)?;
1186
+ defer_drop!(x_val, heap);
1187
+ defer_drop!(y_val, heap);
1188
+
1189
+ let x = value_to_float(x_val, heap)?;
1190
+ let y = value_to_float(y_val, heap)?;
1191
+
1192
+ if y == 0.0 || x.is_infinite() {
1193
+ // CPython raises for both fmod(x, 0) and fmod(inf, y)
1194
+ // but NaN inputs propagate
1195
+ if !x.is_nan() && !y.is_nan() {
1196
+ return Err(math_domain_error());
1197
+ }
1198
+ }
1199
+ Ok(Value::Float(x % y))
1200
+ }
1201
+
1202
+ /// `math.remainder(x, y)` — IEEE 754 remainder of x with respect to y.
1203
+ ///
1204
+ /// The result is `x - n*y` where n is the closest integer to `x/y`.
1205
+ fn math_remainder(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
1206
+ let (x_val, y_val) = args.get_two_args("math.remainder", heap)?;
1207
+ defer_drop!(x_val, heap);
1208
+ defer_drop!(y_val, heap);
1209
+
1210
+ let x = value_to_float(x_val, heap)?;
1211
+ let y = value_to_float(y_val, heap)?;
1212
+
1213
+ // NaN propagates
1214
+ if x.is_nan() || y.is_nan() {
1215
+ return Ok(Value::Float(f64::NAN));
1216
+ }
1217
+ if y == 0.0 {
1218
+ return Err(math_domain_error());
1219
+ }
1220
+ if x.is_infinite() {
1221
+ return Err(math_domain_error());
1222
+ }
1223
+ if y.is_infinite() {
1224
+ return Ok(Value::Float(x));
1225
+ }
1226
+
1227
+ Ok(Value::Float(libm::remainder(x, y)))
1228
+ }
1229
+
1230
+ /// `math.modf(x)` — returns the fractional and integer parts of x as a tuple.
1231
+ ///
1232
+ /// Both values carry the sign of x. Returns `(fractional, integer)`.
1233
+ fn math_modf(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
1234
+ let value = args.get_one_arg("math.modf", heap)?;
1235
+ defer_drop!(value, heap);
1236
+
1237
+ let f = value_to_float(value, heap)?;
1238
+ let (fractional, integer) = libm::modf(f);
1239
+ let tuple = allocate_tuple(smallvec![Value::Float(fractional), Value::Float(integer)], heap)?;
1240
+ Ok(tuple)
1241
+ }
1242
+
1243
+ /// `math.frexp(x)` — returns (mantissa, exponent) such that `x == mantissa * 2**exponent`.
1244
+ ///
1245
+ /// The mantissa is always in the range [0.5, 1.0) or zero.
1246
+ /// Returns a tuple `(float, int)`.
1247
+ fn math_frexp(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
1248
+ let value = args.get_one_arg("math.frexp", heap)?;
1249
+ defer_drop!(value, heap);
1250
+
1251
+ let f = value_to_float(value, heap)?;
1252
+ let (m, exp) = libm::frexp(f);
1253
+ let tuple = allocate_tuple(smallvec![Value::Float(m), Value::Int(i64::from(exp))], heap)?;
1254
+ Ok(tuple)
1255
+ }
1256
+
1257
+ /// `math.ldexp(x, i)` — returns `x * 2**i`, the inverse of `frexp`.
1258
+ ///
1259
+ /// Clamps the exponent to `i32` range before calling `libm::ldexp`, which is safe
1260
+ /// because IEEE 754 double exponents only span -1074 to +1023 — any `i64` outside
1261
+ /// `i32` range would trivially overflow or underflow anyway.
1262
+ fn math_ldexp(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
1263
+ let (x_val, i_val) = args.get_two_args("math.ldexp", heap)?;
1264
+ defer_drop!(x_val, heap);
1265
+ defer_drop!(i_val, heap);
1266
+
1267
+ let x = value_to_float(x_val, heap)?;
1268
+ let i = value_to_int(i_val, heap)?;
1269
+
1270
+ // Special cases: inf/nan/zero pass through regardless of exponent
1271
+ if x.is_nan() || x.is_infinite() || x == 0.0 {
1272
+ return Ok(Value::Float(x));
1273
+ }
1274
+
1275
+ // Clamp i64 to i32 range — exponents beyond ±2 billion trivially overflow/underflow
1276
+ #[expect(clippy::cast_possible_truncation, reason = "clamped to i32 range first")]
1277
+ let exp = i.clamp(i64::from(i32::MIN), i64::from(i32::MAX)) as i32;
1278
+ let result = libm::ldexp(x, exp);
1279
+
1280
+ // If the result overflowed to infinity, CPython raises OverflowError
1281
+ if result.is_infinite() {
1282
+ return Err(math_range_error());
1283
+ }
1284
+
1285
+ Ok(Value::Float(result))
1286
+ }
1287
+
1288
+ // ==========================
1289
+ // Special functions
1290
+ // ==========================
1291
+
1292
+ /// `math.gamma(x)` — returns the Gamma function at x.
1293
+ ///
1294
+ /// CPython 3.14 raises ValueError for non-positive integers:
1295
+ /// "expected a noninteger or positive integer, got <x>".
1296
+ fn math_gamma(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
1297
+ let value = args.get_one_arg("math.gamma", heap)?;
1298
+ defer_drop!(value, heap);
1299
+
1300
+ let f = value_to_float(value, heap)?;
1301
+ // CPython also rejects -inf for gamma (but not lgamma, where lgamma(-inf) = inf)
1302
+ if f == f64::NEG_INFINITY {
1303
+ return Err(SimpleException::new_msg(
1304
+ ExcType::ValueError,
1305
+ format!("expected a noninteger or positive integer, got {f:?}"),
1306
+ )
1307
+ .into());
1308
+ }
1309
+ check_gamma_pole(f)?;
1310
+
1311
+ let result = libm::tgamma(f);
1312
+ check_range_error(result, f)?;
1313
+ Ok(Value::Float(result))
1314
+ }
1315
+
1316
+ /// `math.lgamma(x)` — returns the natural log of the absolute value of Gamma(x).
1317
+ fn math_lgamma(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
1318
+ let value = args.get_one_arg("math.lgamma", heap)?;
1319
+ defer_drop!(value, heap);
1320
+
1321
+ let f = value_to_float(value, heap)?;
1322
+ check_gamma_pole(f)?;
1323
+
1324
+ let result = libm::lgamma(f);
1325
+ check_range_error(result, f)?;
1326
+ Ok(Value::Float(result))
1327
+ }
1328
+
1329
+ /// `math.erf(x)` — returns the error function at x.
1330
+ fn math_erf(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
1331
+ let value = args.get_one_arg("math.erf", heap)?;
1332
+ defer_drop!(value, heap);
1333
+
1334
+ let f = value_to_float(value, heap)?;
1335
+ Ok(Value::Float(libm::erf(f)))
1336
+ }
1337
+
1338
+ /// `math.erfc(x)` — returns the complementary error function at x (1 - erf(x)).
1339
+ ///
1340
+ /// More accurate than `1 - erf(x)` for large x.
1341
+ fn math_erfc(heap: &mut Heap<impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
1342
+ let value = args.get_one_arg("math.erfc", heap)?;
1343
+ defer_drop!(value, heap);
1344
+
1345
+ let f = value_to_float(value, heap)?;
1346
+ Ok(Value::Float(libm::erfc(f)))
1347
+ }
1348
+
1349
+ // ==========================
1350
+ // Helper functions
1351
+ // ==========================
1352
+
1353
+ /// Converts a rounded float to an integer `Value`, checking for infinity/NaN.
1354
+ ///
1355
+ /// `rounded` is the already-rounded float value (e.g., from `floor()`, `ceil()`, `trunc()`).
1356
+ /// `original` is the original input float, used only to determine the error type:
1357
+ /// infinity produces `OverflowError`, NaN produces `ValueError`.
1358
+ ///
1359
+ /// For finite values outside the i64 range, promotes to `LongInt` to match CPython's
1360
+ /// behavior of returning arbitrary-precision integers from `math.floor`/`ceil`/`trunc`.
1361
+ fn float_to_int_checked(rounded: f64, original: f64, heap: &mut Heap<impl ResourceTracker>) -> RunResult<Value> {
1362
+ if original.is_infinite() {
1363
+ Err(SimpleException::new_msg(ExcType::OverflowError, "cannot convert float infinity to integer").into())
1364
+ } else if original.is_nan() {
1365
+ Err(SimpleException::new_msg(ExcType::ValueError, "cannot convert float NaN to integer").into())
1366
+ } else if rounded >= i64::MIN as f64 && rounded < i64::MAX as f64 {
1367
+ // Note: `i64::MAX as f64` rounds up to 2^63 (9223372036854775808.0), so we use
1368
+ // strict less-than to exclude that value. `i64::MIN as f64` is exact (-2^63).
1369
+ #[expect(
1370
+ clippy::cast_possible_truncation,
1371
+ reason = "intentional: value is within i64 range after bounds check"
1372
+ )]
1373
+ let result = rounded as i64;
1374
+ Ok(Value::Int(result))
1375
+ } else {
1376
+ // Value exceeds i64 range — promote to LongInt.
1377
+ // Format with no decimal places and parse as BigInt. This is correct because
1378
+ // `rounded` is already an integer-valued float from floor/ceil/trunc.
1379
+ let s = format!("{rounded:.0}");
1380
+ let bi = s
1381
+ .parse::<BigInt>()
1382
+ .map_err(|_| SimpleException::new_msg(ExcType::ValueError, "float too large to convert to integer"))?;
1383
+ Ok(LongInt::new(bi).into_value(heap)?)
1384
+ }
1385
+ }
1386
+
1387
+ /// Converts a `Value` to `f64`, raising `TypeError` if the value is not numeric.
1388
+ ///
1389
+ /// Accepts `Float`, `Int`, and `Bool` values. For other types, raises a `TypeError`
1390
+ /// with a message matching CPython's format: "must be real number, not <type>".
1391
+ #[expect(
1392
+ clippy::cast_precision_loss,
1393
+ reason = "i64-to-f64 can lose precision for large integers (beyond 2^53), but this matches CPython's conversion semantics"
1394
+ )]
1395
+ fn value_to_float(value: &Value, heap: &Heap<impl ResourceTracker>) -> RunResult<f64> {
1396
+ match value {
1397
+ Value::Float(f) => Ok(*f),
1398
+ Value::Int(n) => Ok(*n as f64),
1399
+ Value::Bool(b) => Ok(if *b { 1.0 } else { 0.0 }),
1400
+ _ => Err(ExcType::type_error(format!(
1401
+ "must be real number, not {}",
1402
+ value.py_type(heap)
1403
+ ))),
1404
+ }
1405
+ }
1406
+
1407
+ /// Converts a `Value` to `i64`, raising `TypeError` if the value is not an integer.
1408
+ ///
1409
+ /// Accepts `Int` and `Bool` values. For other types, raises a `TypeError`
1410
+ /// with a message matching CPython's format.
1411
+ fn value_to_int(value: &Value, heap: &Heap<impl ResourceTracker>) -> RunResult<i64> {
1412
+ match value {
1413
+ Value::Int(n) => Ok(*n),
1414
+ Value::Bool(b) => Ok(i64::from(*b)),
1415
+ _ => Err(ExcType::type_error(format!(
1416
+ "'{}' object cannot be interpreted as an integer",
1417
+ value.py_type(heap)
1418
+ ))),
1419
+ }
1420
+ }
1421
+
1422
+ /// Requires that a float is finite, raising ValueError if it's inf or nan.
1423
+ ///
1424
+ /// CPython 3.14 uses "expected a finite input, got inf" for trig functions.
1425
+ fn require_finite(f: f64) -> RunResult<()> {
1426
+ if f.is_infinite() {
1427
+ Err(SimpleException::new_msg(ExcType::ValueError, format!("expected a finite input, got {f:?}")).into())
1428
+ } else {
1429
+ Ok(())
1430
+ }
1431
+ }
1432
+
1433
+ /// Euclidean GCD algorithm for unsigned 64-bit integers.
1434
+ fn gcd(mut a: u64, mut b: u64) -> u64 {
1435
+ while b != 0 {
1436
+ let t = b;
1437
+ b = a % b;
1438
+ a = t;
1439
+ }
1440
+ a
1441
+ }
1442
+
1443
+ /// Converts a `u64` result to a `Value`, promoting to `LongInt` if it exceeds `i64::MAX`.
1444
+ ///
1445
+ /// This is needed for operations like `gcd(i64::MIN, 0)` where the unsigned result
1446
+ /// (`2^63`) doesn't fit in a signed `i64`.
1447
+ fn u64_to_value(n: u64, heap: &mut Heap<impl ResourceTracker>) -> RunResult<Value> {
1448
+ if let Ok(signed) = i64::try_from(n) {
1449
+ Ok(Value::Int(signed))
1450
+ } else {
1451
+ Ok(LongInt::new(BigInt::from(n)).into_value(heap)?)
1452
+ }
1453
+ }