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.
- package/__tests__/adapter-schema.test.js +2 -0
- package/__tests__/config.test.js +62 -1
- package/__tests__/help-json.test.js +2 -0
- package/__tests__/mcp-adapter.test.js +14 -4
- package/__tests__/mcp-local.test.js +159 -0
- package/__tests__/mcp-stdio-jsonrpc.test.js +105 -0
- package/__tests__/monty-plugin.test.js +121 -0
- package/__tests__/plugin-browser-use-uninstall.test.js +23 -0
- package/__tests__/plugin-browser-use.test.js +77 -0
- package/__tests__/plugins-command.test.js +92 -1
- package/__tests__/plugins-learn.test.js +62 -0
- package/__tests__/plugins-registry.test.js +3 -1
- package/__tests__/resend-plugin.test.js +122 -0
- package/__tests__/skills.test.js +4 -0
- package/cli/adapter-schema.js +3 -2
- package/cli/adapters/mcp.js +22 -3
- package/cli/adapters/process.js +34 -7
- package/cli/config.js +27 -1
- package/cli/help-json.js +2 -2
- package/cli/mcp-diagnostics.js +152 -0
- package/cli/mcp-discovery.js +221 -0
- package/cli/mcp-local.js +267 -25
- package/cli/mcp-stdio-jsonrpc.js +246 -0
- package/cli/plugin-install-guidance.js +25 -0
- package/cli/plugins-command.js +86 -3
- package/cli/plugins-learn.js +177 -0
- package/cli/plugins-manager.js +3 -0
- package/cli/plugins-registry.js +2 -1
- package/cli/skills-mcp.js +102 -0
- package/cli/skills.js +6 -40
- package/cli/supercli.js +7 -2
- package/docs/initial/mcp-local-mode.md +35 -0
- package/docs/mcp-cheatsheet.md +324 -0
- package/docs/plugins.md +7 -0
- package/package.json +1 -1
- package/plugins/browser-use/plugin.json +23 -0
- package/plugins/browser-use/scripts/post-install.js +146 -0
- package/plugins/browser-use/scripts/post-uninstall.js +28 -0
- package/plugins/browser-use/skills/quickstart/SKILL.md +47 -0
- package/plugins/monty/README.md +49 -0
- package/plugins/monty/plugin.json +69 -0
- package/plugins/monty/scripts/post-install.js +73 -0
- package/plugins/monty/scripts/post-uninstall.js +23 -0
- package/plugins/monty/scripts/run-python.js +140 -0
- package/plugins/monty/scripts/setup-monty.js +27 -0
- package/plugins/plugins.json +29 -0
- package/plugins/resend/plugin.json +371 -0
- package/plugins/resend/scripts/post-install.js +59 -0
- package/plugins/resend/scripts/post-uninstall.js +23 -0
- package/plugins/resend/scripts/setup-resend.js +27 -0
- package/plugins/resend/skills/quickstart/SKILL.md +80 -0
- package/ref-monty/.cargo/config.toml +3 -0
- package/ref-monty/.claude/settings.json +60 -0
- package/ref-monty/.claude/skills/fastmod/SKILL.md +22 -0
- package/ref-monty/.claude/skills/python-playground/SKILL.md +47 -0
- package/ref-monty/.codecov.yml +12 -0
- package/ref-monty/.github/actions/build-pgo-wheel/action.yml +72 -0
- package/ref-monty/.github/workflows/ci.yml +776 -0
- package/ref-monty/.github/workflows/codspeed.yml +45 -0
- package/ref-monty/.github/workflows/init-npm-packages.yml +82 -0
- package/ref-monty/.pre-commit-config.yaml +47 -0
- package/ref-monty/.python-version +1 -0
- package/ref-monty/.rustfmt.toml +4 -0
- package/ref-monty/.zed/settings.json +11 -0
- package/ref-monty/CLAUDE.md +535 -0
- package/ref-monty/Cargo.lock +3798 -0
- package/ref-monty/Cargo.toml +87 -0
- package/ref-monty/LICENSE +21 -0
- package/ref-monty/Makefile +216 -0
- package/ref-monty/README.md +430 -0
- package/ref-monty/RELEASING.md +47 -0
- package/ref-monty/crates/fuzz/Cargo.toml +30 -0
- package/ref-monty/crates/fuzz/fuzz_targets/string_input_panic.rs +37 -0
- package/ref-monty/crates/fuzz/fuzz_targets/tokens_input_panic.rs +552 -0
- package/ref-monty/crates/monty/Cargo.toml +68 -0
- package/ref-monty/crates/monty/benches/main.rs +247 -0
- package/ref-monty/crates/monty/build.rs +10 -0
- package/ref-monty/crates/monty/src/args.rs +733 -0
- package/ref-monty/crates/monty/src/asyncio.rs +179 -0
- package/ref-monty/crates/monty/src/builtins/abs.rs +55 -0
- package/ref-monty/crates/monty/src/builtins/all.rs +30 -0
- package/ref-monty/crates/monty/src/builtins/any.rs +30 -0
- package/ref-monty/crates/monty/src/builtins/bin.rs +59 -0
- package/ref-monty/crates/monty/src/builtins/chr.rs +46 -0
- package/ref-monty/crates/monty/src/builtins/divmod.rs +164 -0
- package/ref-monty/crates/monty/src/builtins/enumerate.rs +52 -0
- package/ref-monty/crates/monty/src/builtins/filter.rs +67 -0
- package/ref-monty/crates/monty/src/builtins/getattr.rs +65 -0
- package/ref-monty/crates/monty/src/builtins/hash.rs +28 -0
- package/ref-monty/crates/monty/src/builtins/hex.rs +58 -0
- package/ref-monty/crates/monty/src/builtins/id.rs +24 -0
- package/ref-monty/crates/monty/src/builtins/isinstance.rs +68 -0
- package/ref-monty/crates/monty/src/builtins/len.rs +25 -0
- package/ref-monty/crates/monty/src/builtins/map.rs +98 -0
- package/ref-monty/crates/monty/src/builtins/min_max.rs +113 -0
- package/ref-monty/crates/monty/src/builtins/mod.rs +246 -0
- package/ref-monty/crates/monty/src/builtins/next.rs +21 -0
- package/ref-monty/crates/monty/src/builtins/oct.rs +59 -0
- package/ref-monty/crates/monty/src/builtins/ord.rs +67 -0
- package/ref-monty/crates/monty/src/builtins/pow.rs +365 -0
- package/ref-monty/crates/monty/src/builtins/print.rs +141 -0
- package/ref-monty/crates/monty/src/builtins/repr.rs +16 -0
- package/ref-monty/crates/monty/src/builtins/reversed.rs +28 -0
- package/ref-monty/crates/monty/src/builtins/round.rs +174 -0
- package/ref-monty/crates/monty/src/builtins/sorted.rs +151 -0
- package/ref-monty/crates/monty/src/builtins/sum.rs +66 -0
- package/ref-monty/crates/monty/src/builtins/type_.rs +16 -0
- package/ref-monty/crates/monty/src/builtins/zip.rs +77 -0
- package/ref-monty/crates/monty/src/bytecode/builder.rs +699 -0
- package/ref-monty/crates/monty/src/bytecode/code.rs +310 -0
- package/ref-monty/crates/monty/src/bytecode/compiler.rs +3206 -0
- package/ref-monty/crates/monty/src/bytecode/mod.rs +24 -0
- package/ref-monty/crates/monty/src/bytecode/op.rs +617 -0
- package/ref-monty/crates/monty/src/bytecode/vm/async_exec.rs +1058 -0
- package/ref-monty/crates/monty/src/bytecode/vm/attr.rs +63 -0
- package/ref-monty/crates/monty/src/bytecode/vm/binary.rs +487 -0
- package/ref-monty/crates/monty/src/bytecode/vm/call.rs +767 -0
- package/ref-monty/crates/monty/src/bytecode/vm/collections.rs +741 -0
- package/ref-monty/crates/monty/src/bytecode/vm/compare.rs +147 -0
- package/ref-monty/crates/monty/src/bytecode/vm/exceptions.rs +297 -0
- package/ref-monty/crates/monty/src/bytecode/vm/format.rs +132 -0
- package/ref-monty/crates/monty/src/bytecode/vm/mod.rs +1958 -0
- package/ref-monty/crates/monty/src/bytecode/vm/scheduler.rs +620 -0
- package/ref-monty/crates/monty/src/exception_private.rs +1513 -0
- package/ref-monty/crates/monty/src/exception_public.rs +346 -0
- package/ref-monty/crates/monty/src/expressions.rs +694 -0
- package/ref-monty/crates/monty/src/fstring.rs +854 -0
- package/ref-monty/crates/monty/src/function.rs +119 -0
- package/ref-monty/crates/monty/src/heap.rs +1073 -0
- package/ref-monty/crates/monty/src/heap_data.rs +985 -0
- package/ref-monty/crates/monty/src/heap_traits.rs +312 -0
- package/ref-monty/crates/monty/src/intern.rs +837 -0
- package/ref-monty/crates/monty/src/io.rs +106 -0
- package/ref-monty/crates/monty/src/lib.rs +52 -0
- package/ref-monty/crates/monty/src/modules/asyncio.rs +144 -0
- package/ref-monty/crates/monty/src/modules/math.rs +1453 -0
- package/ref-monty/crates/monty/src/modules/mod.rs +120 -0
- package/ref-monty/crates/monty/src/modules/os.rs +116 -0
- package/ref-monty/crates/monty/src/modules/pathlib.rs +33 -0
- package/ref-monty/crates/monty/src/modules/re.rs +606 -0
- package/ref-monty/crates/monty/src/modules/sys.rs +60 -0
- package/ref-monty/crates/monty/src/modules/typing.rs +70 -0
- package/ref-monty/crates/monty/src/namespace.rs +21 -0
- package/ref-monty/crates/monty/src/object.rs +1040 -0
- package/ref-monty/crates/monty/src/os.rs +215 -0
- package/ref-monty/crates/monty/src/parse.rs +1730 -0
- package/ref-monty/crates/monty/src/prepare.rs +3015 -0
- package/ref-monty/crates/monty/src/repl.rs +1109 -0
- package/ref-monty/crates/monty/src/resource.rs +559 -0
- package/ref-monty/crates/monty/src/run.rs +457 -0
- package/ref-monty/crates/monty/src/run_progress.rs +821 -0
- package/ref-monty/crates/monty/src/signature.rs +651 -0
- package/ref-monty/crates/monty/src/sorting.rs +100 -0
- package/ref-monty/crates/monty/src/types/bytes.rs +2356 -0
- package/ref-monty/crates/monty/src/types/dataclass.rs +345 -0
- package/ref-monty/crates/monty/src/types/dict.rs +879 -0
- package/ref-monty/crates/monty/src/types/dict_view.rs +619 -0
- package/ref-monty/crates/monty/src/types/iter.rs +799 -0
- package/ref-monty/crates/monty/src/types/list.rs +929 -0
- package/ref-monty/crates/monty/src/types/long_int.rs +211 -0
- package/ref-monty/crates/monty/src/types/mod.rs +48 -0
- package/ref-monty/crates/monty/src/types/module.rs +146 -0
- package/ref-monty/crates/monty/src/types/namedtuple.rs +261 -0
- package/ref-monty/crates/monty/src/types/path.rs +596 -0
- package/ref-monty/crates/monty/src/types/property.rs +35 -0
- package/ref-monty/crates/monty/src/types/py_trait.rs +322 -0
- package/ref-monty/crates/monty/src/types/range.rs +285 -0
- package/ref-monty/crates/monty/src/types/re_match.rs +522 -0
- package/ref-monty/crates/monty/src/types/re_pattern.rs +726 -0
- package/ref-monty/crates/monty/src/types/set.rs +1373 -0
- package/ref-monty/crates/monty/src/types/slice.rs +257 -0
- package/ref-monty/crates/monty/src/types/str.rs +2051 -0
- package/ref-monty/crates/monty/src/types/tuple.rs +376 -0
- package/ref-monty/crates/monty/src/types/type.rs +407 -0
- package/ref-monty/crates/monty/src/value.rs +2558 -0
- package/ref-monty/crates/monty/test_cases/args__dict_get_no_args.py +3 -0
- package/ref-monty/crates/monty/test_cases/args__dict_get_too_many.py +3 -0
- package/ref-monty/crates/monty/test_cases/args__dict_items_with_args.py +3 -0
- package/ref-monty/crates/monty/test_cases/args__dict_keys_with_args.py +3 -0
- package/ref-monty/crates/monty/test_cases/args__dict_pop_no_args.py +3 -0
- package/ref-monty/crates/monty/test_cases/args__dict_pop_too_many.py +3 -0
- package/ref-monty/crates/monty/test_cases/args__dict_values_with_args.py +3 -0
- package/ref-monty/crates/monty/test_cases/args__id_too_many.py +2 -0
- package/ref-monty/crates/monty/test_cases/args__len_no_args.py +2 -0
- package/ref-monty/crates/monty/test_cases/args__len_too_many.py +2 -0
- package/ref-monty/crates/monty/test_cases/args__len_type_error_int.py +9 -0
- package/ref-monty/crates/monty/test_cases/args__len_type_error_none.py +9 -0
- package/ref-monty/crates/monty/test_cases/args__list_append_no_args.py +3 -0
- package/ref-monty/crates/monty/test_cases/args__list_append_too_many.py +3 -0
- package/ref-monty/crates/monty/test_cases/args__list_insert_too_few.py +3 -0
- package/ref-monty/crates/monty/test_cases/args__list_insert_too_many.py +3 -0
- package/ref-monty/crates/monty/test_cases/args__repr_no_args.py +2 -0
- package/ref-monty/crates/monty/test_cases/arith__div_zero_float.py +2 -0
- package/ref-monty/crates/monty/test_cases/arith__div_zero_int.py +2 -0
- package/ref-monty/crates/monty/test_cases/arith__floordiv_zero_float.py +2 -0
- package/ref-monty/crates/monty/test_cases/arith__floordiv_zero_int.py +2 -0
- package/ref-monty/crates/monty/test_cases/arith__pow_zero_neg.py +2 -0
- package/ref-monty/crates/monty/test_cases/arith__pow_zero_neg_builtin.py +9 -0
- package/ref-monty/crates/monty/test_cases/assert__expr_fail.py +2 -0
- package/ref-monty/crates/monty/test_cases/assert__fail.py +2 -0
- package/ref-monty/crates/monty/test_cases/assert__fail_msg.py +2 -0
- package/ref-monty/crates/monty/test_cases/assert__fn_fail.py +3 -0
- package/ref-monty/crates/monty/test_cases/assert__ops.py +11 -0
- package/ref-monty/crates/monty/test_cases/async__asyncio_run.py +47 -0
- package/ref-monty/crates/monty/test_cases/async__basic.py +10 -0
- package/ref-monty/crates/monty/test_cases/async__closure.py +14 -0
- package/ref-monty/crates/monty/test_cases/async__double_await_coroutine.py +16 -0
- package/ref-monty/crates/monty/test_cases/async__exception.py +10 -0
- package/ref-monty/crates/monty/test_cases/async__ext_call.py +73 -0
- package/ref-monty/crates/monty/test_cases/async__gather_all.py +85 -0
- package/ref-monty/crates/monty/test_cases/async__nested_await.py +15 -0
- package/ref-monty/crates/monty/test_cases/async__nested_gather_ext.py +37 -0
- package/ref-monty/crates/monty/test_cases/async__not_awaitable.py +10 -0
- package/ref-monty/crates/monty/test_cases/async__not_imported.py +14 -0
- package/ref-monty/crates/monty/test_cases/async__recursion_depth_isolation.py +27 -0
- package/ref-monty/crates/monty/test_cases/async__return_types.py +31 -0
- package/ref-monty/crates/monty/test_cases/async__sequential.py +16 -0
- package/ref-monty/crates/monty/test_cases/async__traceback.py +19 -0
- package/ref-monty/crates/monty/test_cases/async__with_args.py +14 -0
- package/ref-monty/crates/monty/test_cases/attr__get_int_error.py +9 -0
- package/ref-monty/crates/monty/test_cases/attr__get_list_error.py +9 -0
- package/ref-monty/crates/monty/test_cases/attr__set_frozen_nonfield.py +12 -0
- package/ref-monty/crates/monty/test_cases/attr__set_int_error.py +10 -0
- package/ref-monty/crates/monty/test_cases/attr__set_list_error.py +10 -0
- package/ref-monty/crates/monty/test_cases/bench__kitchen_sink.py +68 -0
- package/ref-monty/crates/monty/test_cases/bool__ops.py +20 -0
- package/ref-monty/crates/monty/test_cases/builtin__add_type_error.py +2 -0
- package/ref-monty/crates/monty/test_cases/builtin__filter.py +62 -0
- package/ref-monty/crates/monty/test_cases/builtin__filter_not_iterable.py +11 -0
- package/ref-monty/crates/monty/test_cases/builtin__getattr.py +84 -0
- package/ref-monty/crates/monty/test_cases/builtin__iter_funcs.py +42 -0
- package/ref-monty/crates/monty/test_cases/builtin__iter_next.py +66 -0
- package/ref-monty/crates/monty/test_cases/builtin__map.py +74 -0
- package/ref-monty/crates/monty/test_cases/builtin__map_not_iterable.py +11 -0
- package/ref-monty/crates/monty/test_cases/builtin__math_funcs.py +154 -0
- package/ref-monty/crates/monty/test_cases/builtin__more_iter_funcs.py +148 -0
- package/ref-monty/crates/monty/test_cases/builtin__next_stop_iteration.py +10 -0
- package/ref-monty/crates/monty/test_cases/builtin__print_invalid_kwarg.py +9 -0
- package/ref-monty/crates/monty/test_cases/builtin__print_kwargs.py +12 -0
- package/ref-monty/crates/monty/test_cases/builtin__repr.py +3 -0
- package/ref-monty/crates/monty/test_cases/builtin__string_funcs.py +73 -0
- package/ref-monty/crates/monty/test_cases/bytes__decode_invalid_utf8.py +18 -0
- package/ref-monty/crates/monty/test_cases/bytes__endswith_str_error.py +10 -0
- package/ref-monty/crates/monty/test_cases/bytes__getitem_index_error.py +10 -0
- package/ref-monty/crates/monty/test_cases/bytes__index_start_gt_end.py +10 -0
- package/ref-monty/crates/monty/test_cases/bytes__methods.py +394 -0
- package/ref-monty/crates/monty/test_cases/bytes__negative_count.py +9 -0
- package/ref-monty/crates/monty/test_cases/bytes__ops.py +90 -0
- package/ref-monty/crates/monty/test_cases/bytes__startswith_str_error.py +10 -0
- package/ref-monty/crates/monty/test_cases/call_object.py +3 -0
- package/ref-monty/crates/monty/test_cases/chain_comparison__all.py +79 -0
- package/ref-monty/crates/monty/test_cases/closure__param_shadows_outer.py +81 -0
- package/ref-monty/crates/monty/test_cases/closure__pep448.py +203 -0
- package/ref-monty/crates/monty/test_cases/closure__undefined_nonlocal.py +13 -0
- package/ref-monty/crates/monty/test_cases/compare__mixed_types.py +120 -0
- package/ref-monty/crates/monty/test_cases/comprehension__all.py +208 -0
- package/ref-monty/crates/monty/test_cases/comprehension__scope.py +7 -0
- package/ref-monty/crates/monty/test_cases/comprehension__unbound_local.py +14 -0
- package/ref-monty/crates/monty/test_cases/dataclass__basic.py +238 -0
- package/ref-monty/crates/monty/test_cases/dataclass__call_field_error.py +12 -0
- package/ref-monty/crates/monty/test_cases/dataclass__frozen_set_error.py +12 -0
- package/ref-monty/crates/monty/test_cases/dataclass__get_missing_attr_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/dict__get_unhashable_key.py +3 -0
- package/ref-monty/crates/monty/test_cases/dict__literal_unhashable_key.py +2 -0
- package/ref-monty/crates/monty/test_cases/dict__method_pop_missing_error.py +3 -0
- package/ref-monty/crates/monty/test_cases/dict__methods.py +151 -0
- package/ref-monty/crates/monty/test_cases/dict__ops.py +133 -0
- package/ref-monty/crates/monty/test_cases/dict__pop_unhashable_key.py +4 -0
- package/ref-monty/crates/monty/test_cases/dict__popitem_empty.py +9 -0
- package/ref-monty/crates/monty/test_cases/dict__subscript_missing_key.py +3 -0
- package/ref-monty/crates/monty/test_cases/dict__unhashable_dict_key.py +2 -0
- package/ref-monty/crates/monty/test_cases/dict__unhashable_list_key.py +2 -0
- package/ref-monty/crates/monty/test_cases/dict__unpack_type_error.py +2 -0
- package/ref-monty/crates/monty/test_cases/dict__views.py +165 -0
- package/ref-monty/crates/monty/test_cases/edge__all.py +26 -0
- package/ref-monty/crates/monty/test_cases/edge__float_int_mod.py +2 -0
- package/ref-monty/crates/monty/test_cases/edge__int_float_mod.py +2 -0
- package/ref-monty/crates/monty/test_cases/exc__args.py +16 -0
- package/ref-monty/crates/monty/test_cases/exc__str.py +15 -0
- package/ref-monty/crates/monty/test_cases/execute_ok__all.py +54 -0
- package/ref-monty/crates/monty/test_cases/execute_raise__error_instance_str.py +2 -0
- package/ref-monty/crates/monty/test_cases/execute_raise__error_no_args.py +2 -0
- package/ref-monty/crates/monty/test_cases/execute_raise__error_string_arg.py +2 -0
- package/ref-monty/crates/monty/test_cases/execute_raise__error_string_arg_quotes.py +2 -0
- package/ref-monty/crates/monty/test_cases/execute_raise__error_type.py +2 -0
- package/ref-monty/crates/monty/test_cases/execute_raise__raise_instance_via_var.py +4 -0
- package/ref-monty/crates/monty/test_cases/execute_raise__raise_list.py +2 -0
- package/ref-monty/crates/monty/test_cases/execute_raise__raise_number.py +2 -0
- package/ref-monty/crates/monty/test_cases/execute_raise__raise_type_call_via_var.py +4 -0
- package/ref-monty/crates/monty/test_cases/execute_raise__raise_type_direct.py +3 -0
- package/ref-monty/crates/monty/test_cases/execute_raise__raise_type_via_var.py +4 -0
- package/ref-monty/crates/monty/test_cases/ext_call__arg_side_effect_bug.py +22 -0
- package/ref-monty/crates/monty/test_cases/ext_call__augmented.py +17 -0
- package/ref-monty/crates/monty/test_cases/ext_call__augmented_refcount_bug.py +7 -0
- package/ref-monty/crates/monty/test_cases/ext_call__bare_raise_after_resume.py +34 -0
- package/ref-monty/crates/monty/test_cases/ext_call__basic.py +99 -0
- package/ref-monty/crates/monty/test_cases/ext_call__boolean.py +37 -0
- package/ref-monty/crates/monty/test_cases/ext_call__boolean_side_effect_hang.py +17 -0
- package/ref-monty/crates/monty/test_cases/ext_call__closure_bug.py +16 -0
- package/ref-monty/crates/monty/test_cases/ext_call__comparison.py +26 -0
- package/ref-monty/crates/monty/test_cases/ext_call__deep_call_stack.py +18 -0
- package/ref-monty/crates/monty/test_cases/ext_call__elif.py +171 -0
- package/ref-monty/crates/monty/test_cases/ext_call__exc.py +4 -0
- package/ref-monty/crates/monty/test_cases/ext_call__exc_deep_stack.py +39 -0
- package/ref-monty/crates/monty/test_cases/ext_call__exc_in_function.py +17 -0
- package/ref-monty/crates/monty/test_cases/ext_call__exc_nested_functions.py +31 -0
- package/ref-monty/crates/monty/test_cases/ext_call__ext_exc.py +171 -0
- package/ref-monty/crates/monty/test_cases/ext_call__for.py +114 -0
- package/ref-monty/crates/monty/test_cases/ext_call__fstring.py +12 -0
- package/ref-monty/crates/monty/test_cases/ext_call__if.py +135 -0
- package/ref-monty/crates/monty/test_cases/ext_call__if_condition.py +37 -0
- package/ref-monty/crates/monty/test_cases/ext_call__in_closure.py +14 -0
- package/ref-monty/crates/monty/test_cases/ext_call__in_function.py +40 -0
- package/ref-monty/crates/monty/test_cases/ext_call__in_function_simple.py +7 -0
- package/ref-monty/crates/monty/test_cases/ext_call__literals.py +17 -0
- package/ref-monty/crates/monty/test_cases/ext_call__multi_in_func.py +32 -0
- package/ref-monty/crates/monty/test_cases/ext_call__name_lookup.py +69 -0
- package/ref-monty/crates/monty/test_cases/ext_call__name_lookup_undefined.py +4 -0
- package/ref-monty/crates/monty/test_cases/ext_call__nested_calls.py +14 -0
- package/ref-monty/crates/monty/test_cases/ext_call__recursion_bug.py +19 -0
- package/ref-monty/crates/monty/test_cases/ext_call__return.py +28 -0
- package/ref-monty/crates/monty/test_cases/ext_call__side_effects.py +25 -0
- package/ref-monty/crates/monty/test_cases/ext_call__subscript.py +7 -0
- package/ref-monty/crates/monty/test_cases/ext_call__ternary.py +28 -0
- package/ref-monty/crates/monty/test_cases/ext_call__try.py +280 -0
- package/ref-monty/crates/monty/test_cases/ext_call__try_simple.py +10 -0
- package/ref-monty/crates/monty/test_cases/ext_call__unary.py +13 -0
- package/ref-monty/crates/monty/test_cases/frozenset__ops.py +178 -0
- package/ref-monty/crates/monty/test_cases/fstring__all.py +236 -0
- package/ref-monty/crates/monty/test_cases/fstring__error_eq_align_on_str.py +3 -0
- package/ref-monty/crates/monty/test_cases/fstring__error_float_f_on_str.py +3 -0
- package/ref-monty/crates/monty/test_cases/fstring__error_int_d_on_float.py +3 -0
- package/ref-monty/crates/monty/test_cases/fstring__error_int_d_on_str.py +3 -0
- package/ref-monty/crates/monty/test_cases/fstring__error_invalid_spec.py +4 -0
- package/ref-monty/crates/monty/test_cases/fstring__error_invalid_spec_dynamic.py +4 -0
- package/ref-monty/crates/monty/test_cases/fstring__error_invalid_spec_str.py +4 -0
- package/ref-monty/crates/monty/test_cases/fstring__error_str_s_on_int.py +3 -0
- package/ref-monty/crates/monty/test_cases/function__call_duplicate_kwargs.py +6 -0
- package/ref-monty/crates/monty/test_cases/function__call_unpack.py +42 -0
- package/ref-monty/crates/monty/test_cases/function__defaults.py +117 -0
- package/ref-monty/crates/monty/test_cases/function__err_duplicate_arg.py +7 -0
- package/ref-monty/crates/monty/test_cases/function__err_duplicate_first_arg.py +7 -0
- package/ref-monty/crates/monty/test_cases/function__err_duplicate_kwarg_cleanup.py +9 -0
- package/ref-monty/crates/monty/test_cases/function__err_kwonly_as_positional.py +7 -0
- package/ref-monty/crates/monty/test_cases/function__err_missing_all_posonly.py +7 -0
- package/ref-monty/crates/monty/test_cases/function__err_missing_heap_cleanup.py +9 -0
- package/ref-monty/crates/monty/test_cases/function__err_missing_kwonly.py +7 -0
- package/ref-monty/crates/monty/test_cases/function__err_missing_posonly_with_kwarg.py +7 -0
- package/ref-monty/crates/monty/test_cases/function__err_missing_with_posonly.py +7 -0
- package/ref-monty/crates/monty/test_cases/function__err_posonly_as_kwarg.py +7 -0
- package/ref-monty/crates/monty/test_cases/function__err_posonly_first_as_kwarg.py +7 -0
- package/ref-monty/crates/monty/test_cases/function__err_too_many_posonly.py +7 -0
- package/ref-monty/crates/monty/test_cases/function__err_too_many_with_kwonly.py +7 -0
- package/ref-monty/crates/monty/test_cases/function__err_unexpected_kwarg.py +7 -0
- package/ref-monty/crates/monty/test_cases/function__err_unexpected_kwarg_cleanup.py +9 -0
- package/ref-monty/crates/monty/test_cases/function__err_unexpected_kwarg_quote.py +13 -0
- package/ref-monty/crates/monty/test_cases/function__err_unexpected_kwarg_simple.py +7 -0
- package/ref-monty/crates/monty/test_cases/function__err_unpack_duplicate_arg.py +6 -0
- package/ref-monty/crates/monty/test_cases/function__err_unpack_duplicate_heap.py +8 -0
- package/ref-monty/crates/monty/test_cases/function__err_unpack_int.py +6 -0
- package/ref-monty/crates/monty/test_cases/function__err_unpack_nonstring_key.py +6 -0
- package/ref-monty/crates/monty/test_cases/function__err_unpack_not_mapping.py +6 -0
- package/ref-monty/crates/monty/test_cases/function__kwargs_unpacking.py +173 -0
- package/ref-monty/crates/monty/test_cases/function__ops.py +294 -0
- package/ref-monty/crates/monty/test_cases/function__return_none.py +42 -0
- package/ref-monty/crates/monty/test_cases/function__signatures.py +47 -0
- package/ref-monty/crates/monty/test_cases/function__too_few_args_all.py +6 -0
- package/ref-monty/crates/monty/test_cases/function__too_few_args_one.py +6 -0
- package/ref-monty/crates/monty/test_cases/function__too_few_args_two.py +6 -0
- package/ref-monty/crates/monty/test_cases/function__too_many_args_one.py +6 -0
- package/ref-monty/crates/monty/test_cases/function__too_many_args_two.py +6 -0
- package/ref-monty/crates/monty/test_cases/function__too_many_args_zero.py +6 -0
- package/ref-monty/crates/monty/test_cases/global__error_assigned_before.py +7 -0
- package/ref-monty/crates/monty/test_cases/global__ops.py +163 -0
- package/ref-monty/crates/monty/test_cases/hash__dict_unhashable.py +2 -0
- package/ref-monty/crates/monty/test_cases/hash__list_unhashable.py +2 -0
- package/ref-monty/crates/monty/test_cases/hash__ops.py +153 -0
- package/ref-monty/crates/monty/test_cases/id__bytes_literals_distinct.py +3 -0
- package/ref-monty/crates/monty/test_cases/id__int_copy_distinct.py +5 -0
- package/ref-monty/crates/monty/test_cases/id__is_number_is_number.py +3 -0
- package/ref-monty/crates/monty/test_cases/id__non_overlapping_lifetimes_distinct_types.py +10 -0
- package/ref-monty/crates/monty/test_cases/id__non_overlapping_lifetimes_same_types.py +6 -0
- package/ref-monty/crates/monty/test_cases/id__ops.py +97 -0
- package/ref-monty/crates/monty/test_cases/id__str_literals_same.py +3 -0
- package/ref-monty/crates/monty/test_cases/if__elif_else.py +207 -0
- package/ref-monty/crates/monty/test_cases/if__raise_elif.py +11 -0
- package/ref-monty/crates/monty/test_cases/if__raise_else.py +13 -0
- package/ref-monty/crates/monty/test_cases/if__raise_if.py +9 -0
- package/ref-monty/crates/monty/test_cases/if__raise_in_elif_condition.py +18 -0
- package/ref-monty/crates/monty/test_cases/if__raise_in_if_condition.py +16 -0
- package/ref-monty/crates/monty/test_cases/if_else_expr__all.py +55 -0
- package/ref-monty/crates/monty/test_cases/import__error_cannot_import.py +9 -0
- package/ref-monty/crates/monty/test_cases/import__error_module_not_found.py +9 -0
- package/ref-monty/crates/monty/test_cases/import__local_scope.py +68 -0
- package/ref-monty/crates/monty/test_cases/import__os.py +25 -0
- package/ref-monty/crates/monty/test_cases/import__relative_error.py +9 -0
- package/ref-monty/crates/monty/test_cases/import__relative_no_module_error.py +9 -0
- package/ref-monty/crates/monty/test_cases/import__runtime_error_when_executed.py +14 -0
- package/ref-monty/crates/monty/test_cases/import__star_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/import__sys.py +47 -0
- package/ref-monty/crates/monty/test_cases/import__sys_monty.py +28 -0
- package/ref-monty/crates/monty/test_cases/import__type_checking_guard.py +37 -0
- package/ref-monty/crates/monty/test_cases/import__typing.py +25 -0
- package/ref-monty/crates/monty/test_cases/import__typing_type_ignore.py +4 -0
- package/ref-monty/crates/monty/test_cases/int__bigint.py +467 -0
- package/ref-monty/crates/monty/test_cases/int__bigint_errors.py +260 -0
- package/ref-monty/crates/monty/test_cases/int__ops.py +219 -0
- package/ref-monty/crates/monty/test_cases/int__overflow_division.py +84 -0
- package/ref-monty/crates/monty/test_cases/is_variant__all.py +36 -0
- package/ref-monty/crates/monty/test_cases/isinstance__arg2_list_error.py +2 -0
- package/ref-monty/crates/monty/test_cases/isinstance__arg2_type_error.py +2 -0
- package/ref-monty/crates/monty/test_cases/iter__dict_mutation.py +4 -0
- package/ref-monty/crates/monty/test_cases/iter__for.py +243 -0
- package/ref-monty/crates/monty/test_cases/iter__for_loop_unpacking.py +66 -0
- package/ref-monty/crates/monty/test_cases/iter__generator_expr.py +20 -0
- package/ref-monty/crates/monty/test_cases/iter__generator_expr_type.py +7 -0
- package/ref-monty/crates/monty/test_cases/iter__not_iterable.py +3 -0
- package/ref-monty/crates/monty/test_cases/lambda__all.py +145 -0
- package/ref-monty/crates/monty/test_cases/list__extend_not_iterable.py +7 -0
- package/ref-monty/crates/monty/test_cases/list__getitem_out_of_bounds.py +3 -0
- package/ref-monty/crates/monty/test_cases/list__index_not_found.py +9 -0
- package/ref-monty/crates/monty/test_cases/list__index_start_gt_end.py +10 -0
- package/ref-monty/crates/monty/test_cases/list__ops.py +473 -0
- package/ref-monty/crates/monty/test_cases/list__pop_empty.py +9 -0
- package/ref-monty/crates/monty/test_cases/list__pop_out_of_range.py +9 -0
- package/ref-monty/crates/monty/test_cases/list__pop_type_error.py +9 -0
- package/ref-monty/crates/monty/test_cases/list__remove_not_found.py +9 -0
- package/ref-monty/crates/monty/test_cases/list__setitem_dict_index.py +13 -0
- package/ref-monty/crates/monty/test_cases/list__setitem_huge_int_index.py +13 -0
- package/ref-monty/crates/monty/test_cases/list__setitem_index_error.py +10 -0
- package/ref-monty/crates/monty/test_cases/list__setitem_type_error.py +10 -0
- package/ref-monty/crates/monty/test_cases/list__unpack_type_error.py +2 -0
- package/ref-monty/crates/monty/test_cases/longint__index_error.py +3 -0
- package/ref-monty/crates/monty/test_cases/longint__repeat_error.py +3 -0
- package/ref-monty/crates/monty/test_cases/loop__break_continue.py +113 -0
- package/ref-monty/crates/monty/test_cases/loop__break_finally.py +69 -0
- package/ref-monty/crates/monty/test_cases/loop__break_in_function_error.py +13 -0
- package/ref-monty/crates/monty/test_cases/loop__break_in_if_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/loop__break_nested_except_clears.py +55 -0
- package/ref-monty/crates/monty/test_cases/loop__break_outside_error.py +9 -0
- package/ref-monty/crates/monty/test_cases/loop__continue_finally.py +81 -0
- package/ref-monty/crates/monty/test_cases/loop__continue_in_function_error.py +13 -0
- package/ref-monty/crates/monty/test_cases/loop__continue_in_if_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/loop__continue_nested_except_clears.py +60 -0
- package/ref-monty/crates/monty/test_cases/loop__continue_outside_error.py +9 -0
- package/ref-monty/crates/monty/test_cases/math__acos_domain_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__acosh_domain_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__asin_domain_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__atanh_domain_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__cos_inf_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__cosh_overflow_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__exp_overflow_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__factorial_float_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__factorial_negative_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__floor_inf_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__floor_nan_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__floor_str_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__fmod_inf_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__gamma_neg_int_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__gcd_float_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__isqrt_negative_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__ldexp_overflow_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__log1p_domain_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__log_base1_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__log_zero_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__module.py +1432 -0
- package/ref-monty/crates/monty/test_cases/math__pow_domain_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__sin_inf_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__sqrt_negative_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__tan_inf_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/math__trunc_str_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/method__args_kwargs_unpacking.py +259 -0
- package/ref-monty/crates/monty/test_cases/name_error__unbound_local_func.py +19 -0
- package/ref-monty/crates/monty/test_cases/name_error__unbound_local_module.py +12 -0
- package/ref-monty/crates/monty/test_cases/name_error__undefined_call_chained.py +9 -0
- package/ref-monty/crates/monty/test_cases/name_error__undefined_call_in_expr.py +9 -0
- package/ref-monty/crates/monty/test_cases/name_error__undefined_call_in_function.py +16 -0
- package/ref-monty/crates/monty/test_cases/name_error__undefined_call_with_args.py +9 -0
- package/ref-monty/crates/monty/test_cases/name_error__undefined_global.py +10 -0
- package/ref-monty/crates/monty/test_cases/namedtuple__missing_attr.py +11 -0
- package/ref-monty/crates/monty/test_cases/namedtuple__ops.py +34 -0
- package/ref-monty/crates/monty/test_cases/nonlocal__error_module_level.py +3 -0
- package/ref-monty/crates/monty/test_cases/nonlocal__ops.py +353 -0
- package/ref-monty/crates/monty/test_cases/os__environ.py +40 -0
- package/ref-monty/crates/monty/test_cases/os__getenv_key_list_error.py +5 -0
- package/ref-monty/crates/monty/test_cases/os__getenv_key_type_error.py +5 -0
- package/ref-monty/crates/monty/test_cases/parse_error__complex.py +3 -0
- package/ref-monty/crates/monty/test_cases/pathlib__import.py +11 -0
- package/ref-monty/crates/monty/test_cases/pathlib__os.py +136 -0
- package/ref-monty/crates/monty/test_cases/pathlib__os_read_error.py +12 -0
- package/ref-monty/crates/monty/test_cases/pathlib__pure.py +81 -0
- package/ref-monty/crates/monty/test_cases/pyobject__cycle_dict_self.py +5 -0
- package/ref-monty/crates/monty/test_cases/pyobject__cycle_list_dict.py +6 -0
- package/ref-monty/crates/monty/test_cases/pyobject__cycle_list_self.py +5 -0
- package/ref-monty/crates/monty/test_cases/pyobject__cycle_multiple_refs.py +6 -0
- package/ref-monty/crates/monty/test_cases/range__error_no_args.py +2 -0
- package/ref-monty/crates/monty/test_cases/range__error_step_zero.py +2 -0
- package/ref-monty/crates/monty/test_cases/range__error_too_many_args.py +2 -0
- package/ref-monty/crates/monty/test_cases/range__getitem_index_error.py +10 -0
- package/ref-monty/crates/monty/test_cases/range__ops.py +236 -0
- package/ref-monty/crates/monty/test_cases/re__basic.py +756 -0
- package/ref-monty/crates/monty/test_cases/re__grouping.py +241 -0
- package/ref-monty/crates/monty/test_cases/re__match.py +148 -0
- package/ref-monty/crates/monty/test_cases/recursion__deep_drop.py +26 -0
- package/ref-monty/crates/monty/test_cases/recursion__deep_eq.py +23 -0
- package/ref-monty/crates/monty/test_cases/recursion__deep_hash.py +46 -0
- package/ref-monty/crates/monty/test_cases/recursion__deep_repr.py +12 -0
- package/ref-monty/crates/monty/test_cases/recursion__function_depth.py +13 -0
- package/ref-monty/crates/monty/test_cases/refcount__cycle_mutual_reference.py +18 -0
- package/ref-monty/crates/monty/test_cases/refcount__cycle_self_reference.py +12 -0
- package/ref-monty/crates/monty/test_cases/refcount__dict_basic.py +5 -0
- package/ref-monty/crates/monty/test_cases/refcount__dict_get.py +5 -0
- package/ref-monty/crates/monty/test_cases/refcount__dict_keys_and.py +14 -0
- package/ref-monty/crates/monty/test_cases/refcount__dict_overwrite.py +6 -0
- package/ref-monty/crates/monty/test_cases/refcount__gather_cleanup.py +16 -0
- package/ref-monty/crates/monty/test_cases/refcount__gather_exception.py +18 -0
- package/ref-monty/crates/monty/test_cases/refcount__gather_nested_cancel.py +25 -0
- package/ref-monty/crates/monty/test_cases/refcount__immediate_skipped.py +4 -0
- package/ref-monty/crates/monty/test_cases/refcount__kwargs_unpacking.py +27 -0
- package/ref-monty/crates/monty/test_cases/refcount__list_append_multiple.py +6 -0
- package/ref-monty/crates/monty/test_cases/refcount__list_append_ref.py +5 -0
- package/ref-monty/crates/monty/test_cases/refcount__list_concat.py +5 -0
- package/ref-monty/crates/monty/test_cases/refcount__list_getitem.py +5 -0
- package/ref-monty/crates/monty/test_cases/refcount__list_iadd.py +5 -0
- package/ref-monty/crates/monty/test_cases/refcount__nested_list.py +4 -0
- package/ref-monty/crates/monty/test_cases/refcount__re_pattern_sub_error_paths.py +37 -0
- package/ref-monty/crates/monty/test_cases/refcount__re_search_match.py +34 -0
- package/ref-monty/crates/monty/test_cases/refcount__re_sub_error_paths.py +31 -0
- package/ref-monty/crates/monty/test_cases/refcount__shared_reference.py +4 -0
- package/ref-monty/crates/monty/test_cases/refcount__single_list.py +3 -0
- package/ref-monty/crates/monty/test_cases/repr__cycle_detection.py +24 -0
- package/ref-monty/crates/monty/test_cases/set__ops.py +191 -0
- package/ref-monty/crates/monty/test_cases/set__review_bugs.py +35 -0
- package/ref-monty/crates/monty/test_cases/set__unpack_type_error.py +2 -0
- package/ref-monty/crates/monty/test_cases/slice__invalid_indices.py +2 -0
- package/ref-monty/crates/monty/test_cases/slice__kwargs.py +9 -0
- package/ref-monty/crates/monty/test_cases/slice__no_args.py +9 -0
- package/ref-monty/crates/monty/test_cases/slice__ops.py +149 -0
- package/ref-monty/crates/monty/test_cases/slice__step_zero.py +9 -0
- package/ref-monty/crates/monty/test_cases/slice__step_zero_bytes.py +9 -0
- package/ref-monty/crates/monty/test_cases/slice__step_zero_range.py +9 -0
- package/ref-monty/crates/monty/test_cases/slice__step_zero_str.py +9 -0
- package/ref-monty/crates/monty/test_cases/slice__step_zero_tuple.py +9 -0
- package/ref-monty/crates/monty/test_cases/slice__too_many_args.py +9 -0
- package/ref-monty/crates/monty/test_cases/str__getitem_index_error.py +10 -0
- package/ref-monty/crates/monty/test_cases/str__index_not_found.py +9 -0
- package/ref-monty/crates/monty/test_cases/str__join_no_args.py +9 -0
- package/ref-monty/crates/monty/test_cases/str__join_non_string.py +9 -0
- package/ref-monty/crates/monty/test_cases/str__join_not_iterable.py +9 -0
- package/ref-monty/crates/monty/test_cases/str__join_too_many_args.py +9 -0
- package/ref-monty/crates/monty/test_cases/str__methods.py +327 -0
- package/ref-monty/crates/monty/test_cases/str__ops.py +162 -0
- package/ref-monty/crates/monty/test_cases/str__partition_empty.py +9 -0
- package/ref-monty/crates/monty/test_cases/str__rsplit_empty_sep.py +9 -0
- package/ref-monty/crates/monty/test_cases/str__split_empty_sep.py +9 -0
- package/ref-monty/crates/monty/test_cases/sys__types.py +7 -0
- package/ref-monty/crates/monty/test_cases/traceback__division_error.py +30 -0
- package/ref-monty/crates/monty/test_cases/traceback__index_error.py +17 -0
- package/ref-monty/crates/monty/test_cases/traceback__insert_as_int.py +10 -0
- package/ref-monty/crates/monty/test_cases/traceback__nested_call.py +29 -0
- package/ref-monty/crates/monty/test_cases/traceback__nonlocal_module_scope.py +10 -0
- package/ref-monty/crates/monty/test_cases/traceback__nonlocal_unbound.py +24 -0
- package/ref-monty/crates/monty/test_cases/traceback__range_as_int.py +9 -0
- package/ref-monty/crates/monty/test_cases/traceback__recursion_error.py +23 -0
- package/ref-monty/crates/monty/test_cases/traceback__set_mutation.py +11 -0
- package/ref-monty/crates/monty/test_cases/traceback__undefined_attr_call.py +16 -0
- package/ref-monty/crates/monty/test_cases/traceback__undefined_call.py +16 -0
- package/ref-monty/crates/monty/test_cases/traceback__undefined_raise.py +16 -0
- package/ref-monty/crates/monty/test_cases/try_except__all.py +472 -0
- package/ref-monty/crates/monty/test_cases/try_except__bare_raise_no_context.py +2 -0
- package/ref-monty/crates/monty/test_cases/try_except__invalid_type.py +5 -0
- package/ref-monty/crates/monty/test_cases/tuple__getitem_out_of_bounds.py +3 -0
- package/ref-monty/crates/monty/test_cases/tuple__index_not_found.py +9 -0
- package/ref-monty/crates/monty/test_cases/tuple__index_start_gt_end.py +10 -0
- package/ref-monty/crates/monty/test_cases/tuple__methods.py +19 -0
- package/ref-monty/crates/monty/test_cases/tuple__ops.py +133 -0
- package/ref-monty/crates/monty/test_cases/tuple__unpack_type_error.py +2 -0
- package/ref-monty/crates/monty/test_cases/type__builtin_attr_error.py +9 -0
- package/ref-monty/crates/monty/test_cases/type__bytes_negative.py +2 -0
- package/ref-monty/crates/monty/test_cases/type__cell_not_builtin.py +9 -0
- package/ref-monty/crates/monty/test_cases/type__exception_attr_error.py +11 -0
- package/ref-monty/crates/monty/test_cases/type__float_conversion_error.py +2 -0
- package/ref-monty/crates/monty/test_cases/type__float_repr_both_quotes.py +9 -0
- package/ref-monty/crates/monty/test_cases/type__float_repr_newline.py +9 -0
- package/ref-monty/crates/monty/test_cases/type__float_repr_single_quote.py +9 -0
- package/ref-monty/crates/monty/test_cases/type__int_conversion_error.py +2 -0
- package/ref-monty/crates/monty/test_cases/type__list_not_iterable.py +2 -0
- package/ref-monty/crates/monty/test_cases/type__non_builtin_name_error.py +9 -0
- package/ref-monty/crates/monty/test_cases/type__ops.py +200 -0
- package/ref-monty/crates/monty/test_cases/type__shadow_exc.py +3 -0
- package/ref-monty/crates/monty/test_cases/type__shadow_int.py +9 -0
- package/ref-monty/crates/monty/test_cases/type__shadow_len.py +3 -0
- package/ref-monty/crates/monty/test_cases/type__tuple_not_iterable.py +2 -0
- package/ref-monty/crates/monty/test_cases/type_error__int_add_list.py +2 -0
- package/ref-monty/crates/monty/test_cases/type_error__int_div_str.py +2 -0
- package/ref-monty/crates/monty/test_cases/type_error__int_floordiv_str.py +2 -0
- package/ref-monty/crates/monty/test_cases/type_error__int_iadd_str.py +3 -0
- package/ref-monty/crates/monty/test_cases/type_error__int_mod_str.py +2 -0
- package/ref-monty/crates/monty/test_cases/type_error__int_pow_str.py +2 -0
- package/ref-monty/crates/monty/test_cases/type_error__int_sub_str.py +2 -0
- package/ref-monty/crates/monty/test_cases/type_error__list_add_int.py +2 -0
- package/ref-monty/crates/monty/test_cases/type_error__list_add_str.py +2 -0
- package/ref-monty/crates/monty/test_cases/type_error__list_iadd_int.py +6 -0
- package/ref-monty/crates/monty/test_cases/type_error__str_add_int.py +2 -0
- package/ref-monty/crates/monty/test_cases/type_error__str_iadd_int.py +3 -0
- package/ref-monty/crates/monty/test_cases/type_error__unary_invert_str.py +3 -0
- package/ref-monty/crates/monty/test_cases/type_error__unary_minus_str.py +4 -0
- package/ref-monty/crates/monty/test_cases/type_error__unary_neg_str.py +3 -0
- package/ref-monty/crates/monty/test_cases/type_error__unary_plus_str.py +4 -0
- package/ref-monty/crates/monty/test_cases/typing__types.py +24 -0
- package/ref-monty/crates/monty/test_cases/unpack__nested.py +48 -0
- package/ref-monty/crates/monty/test_cases/unpack__non_sequence.py +9 -0
- package/ref-monty/crates/monty/test_cases/unpack__not_enough.py +9 -0
- package/ref-monty/crates/monty/test_cases/unpack__ops.py +153 -0
- package/ref-monty/crates/monty/test_cases/unpack__star_not_enough.py +9 -0
- package/ref-monty/crates/monty/test_cases/unpack__too_many.py +9 -0
- package/ref-monty/crates/monty/test_cases/version__cpython.py +4 -0
- package/ref-monty/crates/monty/test_cases/walrus__all.py +178 -0
- package/ref-monty/crates/monty/test_cases/while__all.py +206 -0
- package/ref-monty/crates/monty/tests/asyncio.rs +764 -0
- package/ref-monty/crates/monty/tests/binary_serde.rs +185 -0
- package/ref-monty/crates/monty/tests/bytecode_limits.rs +248 -0
- package/ref-monty/crates/monty/tests/datatest_runner.rs +2029 -0
- package/ref-monty/crates/monty/tests/inputs.rs +420 -0
- package/ref-monty/crates/monty/tests/json_serde.rs +250 -0
- package/ref-monty/crates/monty/tests/main.rs +71 -0
- package/ref-monty/crates/monty/tests/math_module.rs +114 -0
- package/ref-monty/crates/monty/tests/name_lookup.rs +482 -0
- package/ref-monty/crates/monty/tests/os_tests.rs +459 -0
- package/ref-monty/crates/monty/tests/parse_errors.rs +441 -0
- package/ref-monty/crates/monty/tests/print_writer.rs +238 -0
- package/ref-monty/crates/monty/tests/py_object.rs +121 -0
- package/ref-monty/crates/monty/tests/regex.rs +90 -0
- package/ref-monty/crates/monty/tests/repl.rs +344 -0
- package/ref-monty/crates/monty/tests/resource_limits.rs +1826 -0
- package/ref-monty/crates/monty/tests/try_from.rs +167 -0
- package/ref-monty/crates/monty-cli/Cargo.toml +25 -0
- package/ref-monty/crates/monty-cli/src/main.rs +541 -0
- package/ref-monty/crates/monty-js/.cargo/config.toml +2 -0
- package/ref-monty/crates/monty-js/.prettierignore +8 -0
- package/ref-monty/crates/monty-js/Cargo.toml +32 -0
- package/ref-monty/crates/monty-js/README.md +207 -0
- package/ref-monty/crates/monty-js/__test__/async.spec.ts +350 -0
- package/ref-monty/crates/monty-js/__test__/basic.spec.ts +114 -0
- package/ref-monty/crates/monty-js/__test__/exceptions.spec.ts +427 -0
- package/ref-monty/crates/monty-js/__test__/external.spec.ts +354 -0
- package/ref-monty/crates/monty-js/__test__/inputs.spec.ts +143 -0
- package/ref-monty/crates/monty-js/__test__/limits.spec.ts +162 -0
- package/ref-monty/crates/monty-js/__test__/package.json +3 -0
- package/ref-monty/crates/monty-js/__test__/print.spec.ts +229 -0
- package/ref-monty/crates/monty-js/__test__/repl.spec.ts +34 -0
- package/ref-monty/crates/monty-js/__test__/serialize.spec.ts +205 -0
- package/ref-monty/crates/monty-js/__test__/start.spec.ts +443 -0
- package/ref-monty/crates/monty-js/__test__/type_check.spec.ts +147 -0
- package/ref-monty/crates/monty-js/__test__/types.spec.ts +319 -0
- package/ref-monty/crates/monty-js/build.rs +61 -0
- package/ref-monty/crates/monty-js/index-header.d.ts +3 -0
- package/ref-monty/crates/monty-js/package-lock.json +4694 -0
- package/ref-monty/crates/monty-js/package.json +100 -0
- package/ref-monty/crates/monty-js/scripts/smoke-test.sh +69 -0
- package/ref-monty/crates/monty-js/smoke-test/package.json +17 -0
- package/ref-monty/crates/monty-js/smoke-test/test.ts +171 -0
- package/ref-monty/crates/monty-js/smoke-test/tsconfig.json +11 -0
- package/ref-monty/crates/monty-js/src/convert.rs +648 -0
- package/ref-monty/crates/monty-js/src/exceptions.rs +293 -0
- package/ref-monty/crates/monty-js/src/lib.rs +41 -0
- package/ref-monty/crates/monty-js/src/limits.rs +53 -0
- package/ref-monty/crates/monty-js/src/monty_cls.rs +1407 -0
- package/ref-monty/crates/monty-js/tsconfig.json +17 -0
- package/ref-monty/crates/monty-js/wrapper.ts +701 -0
- package/ref-monty/crates/monty-python/Cargo.toml +38 -0
- package/ref-monty/crates/monty-python/README.md +134 -0
- package/ref-monty/crates/monty-python/build.rs +4 -0
- package/ref-monty/crates/monty-python/example.py +40 -0
- package/ref-monty/crates/monty-python/exercise.py +46 -0
- package/ref-monty/crates/monty-python/pyproject.toml +57 -0
- package/ref-monty/crates/monty-python/python/pydantic_monty/__init__.py +281 -0
- package/ref-monty/crates/monty-python/python/pydantic_monty/_monty.pyi +677 -0
- package/ref-monty/crates/monty-python/python/pydantic_monty/os_access.py +933 -0
- package/ref-monty/crates/monty-python/python/pydantic_monty/py.typed +0 -0
- package/ref-monty/crates/monty-python/src/convert.rs +273 -0
- package/ref-monty/crates/monty-python/src/dataclass.rs +461 -0
- package/ref-monty/crates/monty-python/src/exceptions.rs +557 -0
- package/ref-monty/crates/monty-python/src/external.rs +165 -0
- package/ref-monty/crates/monty-python/src/lib.rs +77 -0
- package/ref-monty/crates/monty-python/src/limits.rs +142 -0
- package/ref-monty/crates/monty-python/src/monty_cls.rs +1650 -0
- package/ref-monty/crates/monty-python/src/repl.rs +470 -0
- package/ref-monty/crates/monty-python/src/serialization.rs +761 -0
- package/ref-monty/crates/monty-python/tests/test_async.py +1201 -0
- package/ref-monty/crates/monty-python/tests/test_basic.py +66 -0
- package/ref-monty/crates/monty-python/tests/test_dataclasses.py +971 -0
- package/ref-monty/crates/monty-python/tests/test_exceptions.py +361 -0
- package/ref-monty/crates/monty-python/tests/test_external.py +367 -0
- package/ref-monty/crates/monty-python/tests/test_inputs.py +126 -0
- package/ref-monty/crates/monty-python/tests/test_limits.py +257 -0
- package/ref-monty/crates/monty-python/tests/test_os_access.py +1286 -0
- package/ref-monty/crates/monty-python/tests/test_os_access_compat.py +731 -0
- package/ref-monty/crates/monty-python/tests/test_os_access_raw.py +483 -0
- package/ref-monty/crates/monty-python/tests/test_os_calls.py +819 -0
- package/ref-monty/crates/monty-python/tests/test_print.py +208 -0
- package/ref-monty/crates/monty-python/tests/test_re.py +170 -0
- package/ref-monty/crates/monty-python/tests/test_readme_examples.py +20 -0
- package/ref-monty/crates/monty-python/tests/test_repl.py +749 -0
- package/ref-monty/crates/monty-python/tests/test_serialize.py +284 -0
- package/ref-monty/crates/monty-python/tests/test_start.py +346 -0
- package/ref-monty/crates/monty-python/tests/test_threading.py +163 -0
- package/ref-monty/crates/monty-python/tests/test_type_check.py +344 -0
- package/ref-monty/crates/monty-python/tests/test_types.py +553 -0
- package/ref-monty/crates/monty-type-checking/Cargo.toml +32 -0
- package/ref-monty/crates/monty-type-checking/src/db.rs +116 -0
- package/ref-monty/crates/monty-type-checking/src/lib.rs +4 -0
- package/ref-monty/crates/monty-type-checking/src/type_check.rs +280 -0
- package/ref-monty/crates/monty-type-checking/tests/bad_types.py +109 -0
- package/ref-monty/crates/monty-type-checking/tests/bad_types_output.txt +21 -0
- package/ref-monty/crates/monty-type-checking/tests/good_types.py +475 -0
- package/ref-monty/crates/monty-type-checking/tests/main.rs +205 -0
- package/ref-monty/crates/monty-type-checking/tests/reveal_types.py +56 -0
- package/ref-monty/crates/monty-type-checking/tests/reveal_types_output.txt +41 -0
- package/ref-monty/crates/monty-typeshed/Cargo.toml +29 -0
- package/ref-monty/crates/monty-typeshed/README.md +11 -0
- package/ref-monty/crates/monty-typeshed/build.rs +101 -0
- package/ref-monty/crates/monty-typeshed/custom/README.md +1 -0
- package/ref-monty/crates/monty-typeshed/custom/asyncio.pyi +138 -0
- package/ref-monty/crates/monty-typeshed/custom/os.pyi +87 -0
- package/ref-monty/crates/monty-typeshed/custom/sys.pyi +33 -0
- package/ref-monty/crates/monty-typeshed/src/lib.rs +56 -0
- package/ref-monty/crates/monty-typeshed/update.py +321 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/source_commit.txt +1 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/VERSIONS +20 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/_collections_abc.pyi +105 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/_typeshed/__init__.pyi +394 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/asyncio.pyi +138 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/builtins.pyi +1434 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/collections/__init__.pyi +527 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/collections/abc.pyi +2 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/dataclasses.pyi +502 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/enum.pyi +376 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/math.pyi +149 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/os.pyi +87 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/pathlib/__init__.pyi +395 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/pathlib/types.pyi +8 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/re.pyi +337 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/sys.pyi +33 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/types.pyi +741 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/typing.pyi +1217 -0
- package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/typing_extensions.pyi +716 -0
- package/ref-monty/docs/usage-guide.md +117 -0
- package/ref-monty/examples/README.md +3 -0
- package/ref-monty/examples/expense_analysis/README.md +3 -0
- package/ref-monty/examples/expense_analysis/data.py +124 -0
- package/ref-monty/examples/expense_analysis/main.py +115 -0
- package/ref-monty/examples/sql_playground/README.md +20 -0
- package/ref-monty/examples/sql_playground/external_functions.py +129 -0
- package/ref-monty/examples/sql_playground/main.py +81 -0
- package/ref-monty/examples/sql_playground/sandbox_code.py +82 -0
- package/ref-monty/examples/sql_playground/type_stubs.pyi +14 -0
- package/ref-monty/examples/web_scraper/README.md +15 -0
- package/ref-monty/examples/web_scraper/browser.py +56 -0
- package/ref-monty/examples/web_scraper/example_code.py +59 -0
- package/ref-monty/examples/web_scraper/external_functions.py +324 -0
- package/ref-monty/examples/web_scraper/main.py +193 -0
- package/ref-monty/examples/web_scraper/sub_agent.py +79 -0
- package/ref-monty/monty-npm.md +235 -0
- package/ref-monty/pyproject.toml +162 -0
- package/ref-monty/scripts/check_imports.py +91 -0
- package/ref-monty/scripts/codecov_diff.py +412 -0
- package/ref-monty/scripts/complete_tests.py +146 -0
- package/ref-monty/scripts/flamegraph_to_text.py +208 -0
- package/ref-monty/scripts/iter_test_methods.py +540 -0
- package/ref-monty/scripts/run_traceback.py +180 -0
- package/ref-monty/scripts/startup_performance.py +130 -0
- package/ref-monty/uv.lock +1779 -0
- package/temp_resend_cli/repo/.github/scripts/pr-title-check.js +34 -0
- package/temp_resend_cli/repo/.github/workflows/ci.yml +67 -0
- package/temp_resend_cli/repo/.github/workflows/post-release.yml +51 -0
- package/temp_resend_cli/repo/.github/workflows/pr-title-check.yml +13 -0
- package/temp_resend_cli/repo/.github/workflows/release.yml +175 -0
- package/temp_resend_cli/repo/.github/workflows/test-install-unix.yml +34 -0
- package/temp_resend_cli/repo/.github/workflows/test-install-windows.yml +48 -0
- package/temp_resend_cli/repo/CHANGELOG.md +31 -0
- package/temp_resend_cli/repo/LICENSE +21 -0
- package/temp_resend_cli/repo/README.md +450 -0
- package/temp_resend_cli/repo/biome.json +36 -0
- package/temp_resend_cli/repo/install.ps1 +141 -0
- package/temp_resend_cli/repo/install.sh +301 -0
- package/temp_resend_cli/repo/package.json +61 -0
- package/temp_resend_cli/repo/pnpm-lock.yaml +2439 -0
- package/temp_resend_cli/repo/renovate.json +4 -0
- package/temp_resend_cli/repo/src/cli.ts +98 -0
- package/temp_resend_cli/repo/src/commands/api-keys/create.ts +114 -0
- package/temp_resend_cli/repo/src/commands/api-keys/delete.ts +47 -0
- package/temp_resend_cli/repo/src/commands/api-keys/index.ts +26 -0
- package/temp_resend_cli/repo/src/commands/api-keys/list.ts +35 -0
- package/temp_resend_cli/repo/src/commands/api-keys/utils.ts +8 -0
- package/temp_resend_cli/repo/src/commands/auth/index.ts +20 -0
- package/temp_resend_cli/repo/src/commands/auth/login.ts +234 -0
- package/temp_resend_cli/repo/src/commands/auth/logout.ts +105 -0
- package/temp_resend_cli/repo/src/commands/broadcasts/create.ts +196 -0
- package/temp_resend_cli/repo/src/commands/broadcasts/delete.ts +46 -0
- package/temp_resend_cli/repo/src/commands/broadcasts/get.ts +59 -0
- package/temp_resend_cli/repo/src/commands/broadcasts/index.ts +43 -0
- package/temp_resend_cli/repo/src/commands/broadcasts/list.ts +60 -0
- package/temp_resend_cli/repo/src/commands/broadcasts/send.ts +56 -0
- package/temp_resend_cli/repo/src/commands/broadcasts/update.ts +95 -0
- package/temp_resend_cli/repo/src/commands/broadcasts/utils.ts +35 -0
- package/temp_resend_cli/repo/src/commands/contact-properties/create.ts +118 -0
- package/temp_resend_cli/repo/src/commands/contact-properties/delete.ts +48 -0
- package/temp_resend_cli/repo/src/commands/contact-properties/get.ts +46 -0
- package/temp_resend_cli/repo/src/commands/contact-properties/index.ts +48 -0
- package/temp_resend_cli/repo/src/commands/contact-properties/list.ts +68 -0
- package/temp_resend_cli/repo/src/commands/contact-properties/update.ts +88 -0
- package/temp_resend_cli/repo/src/commands/contact-properties/utils.ts +17 -0
- package/temp_resend_cli/repo/src/commands/contacts/add-segment.ts +78 -0
- package/temp_resend_cli/repo/src/commands/contacts/create.ts +122 -0
- package/temp_resend_cli/repo/src/commands/contacts/delete.ts +49 -0
- package/temp_resend_cli/repo/src/commands/contacts/get.ts +53 -0
- package/temp_resend_cli/repo/src/commands/contacts/index.ts +58 -0
- package/temp_resend_cli/repo/src/commands/contacts/list.ts +57 -0
- package/temp_resend_cli/repo/src/commands/contacts/remove-segment.ts +48 -0
- package/temp_resend_cli/repo/src/commands/contacts/segments.ts +39 -0
- package/temp_resend_cli/repo/src/commands/contacts/topics.ts +45 -0
- package/temp_resend_cli/repo/src/commands/contacts/update-topics.ts +90 -0
- package/temp_resend_cli/repo/src/commands/contacts/update.ts +77 -0
- package/temp_resend_cli/repo/src/commands/contacts/utils.ts +119 -0
- package/temp_resend_cli/repo/src/commands/doctor.ts +216 -0
- package/temp_resend_cli/repo/src/commands/domains/create.ts +83 -0
- package/temp_resend_cli/repo/src/commands/domains/delete.ts +42 -0
- package/temp_resend_cli/repo/src/commands/domains/get.ts +47 -0
- package/temp_resend_cli/repo/src/commands/domains/index.ts +35 -0
- package/temp_resend_cli/repo/src/commands/domains/list.ts +53 -0
- package/temp_resend_cli/repo/src/commands/domains/update.ts +75 -0
- package/temp_resend_cli/repo/src/commands/domains/utils.ts +44 -0
- package/temp_resend_cli/repo/src/commands/domains/verify.ts +38 -0
- package/temp_resend_cli/repo/src/commands/emails/batch.ts +140 -0
- package/temp_resend_cli/repo/src/commands/emails/get.ts +44 -0
- package/temp_resend_cli/repo/src/commands/emails/index.ts +30 -0
- package/temp_resend_cli/repo/src/commands/emails/list.ts +84 -0
- package/temp_resend_cli/repo/src/commands/emails/receiving/attachment.ts +55 -0
- package/temp_resend_cli/repo/src/commands/emails/receiving/attachments.ts +68 -0
- package/temp_resend_cli/repo/src/commands/emails/receiving/get.ts +58 -0
- package/temp_resend_cli/repo/src/commands/emails/receiving/index.ts +28 -0
- package/temp_resend_cli/repo/src/commands/emails/receiving/list.ts +59 -0
- package/temp_resend_cli/repo/src/commands/emails/receiving/utils.ts +38 -0
- package/temp_resend_cli/repo/src/commands/emails/send.ts +189 -0
- package/temp_resend_cli/repo/src/commands/open.ts +27 -0
- package/temp_resend_cli/repo/src/commands/segments/create.ts +50 -0
- package/temp_resend_cli/repo/src/commands/segments/delete.ts +47 -0
- package/temp_resend_cli/repo/src/commands/segments/get.ts +38 -0
- package/temp_resend_cli/repo/src/commands/segments/index.ts +36 -0
- package/temp_resend_cli/repo/src/commands/segments/list.ts +58 -0
- package/temp_resend_cli/repo/src/commands/segments/utils.ts +7 -0
- package/temp_resend_cli/repo/src/commands/teams/index.ts +10 -0
- package/temp_resend_cli/repo/src/commands/teams/list.ts +35 -0
- package/temp_resend_cli/repo/src/commands/teams/remove.ts +86 -0
- package/temp_resend_cli/repo/src/commands/teams/switch.ts +76 -0
- package/temp_resend_cli/repo/src/commands/topics/create.ts +73 -0
- package/temp_resend_cli/repo/src/commands/topics/delete.ts +47 -0
- package/temp_resend_cli/repo/src/commands/topics/get.ts +42 -0
- package/temp_resend_cli/repo/src/commands/topics/index.ts +42 -0
- package/temp_resend_cli/repo/src/commands/topics/list.ts +34 -0
- package/temp_resend_cli/repo/src/commands/topics/update.ts +59 -0
- package/temp_resend_cli/repo/src/commands/topics/utils.ts +16 -0
- package/temp_resend_cli/repo/src/commands/webhooks/create.ts +128 -0
- package/temp_resend_cli/repo/src/commands/webhooks/delete.ts +49 -0
- package/temp_resend_cli/repo/src/commands/webhooks/get.ts +42 -0
- package/temp_resend_cli/repo/src/commands/webhooks/index.ts +42 -0
- package/temp_resend_cli/repo/src/commands/webhooks/list.ts +55 -0
- package/temp_resend_cli/repo/src/commands/webhooks/listen.ts +379 -0
- package/temp_resend_cli/repo/src/commands/webhooks/update.ts +83 -0
- package/temp_resend_cli/repo/src/commands/webhooks/utils.ts +36 -0
- package/temp_resend_cli/repo/src/commands/whoami.ts +71 -0
- package/temp_resend_cli/repo/src/lib/actions.ts +157 -0
- package/temp_resend_cli/repo/src/lib/client.ts +37 -0
- package/temp_resend_cli/repo/src/lib/config.ts +217 -0
- package/temp_resend_cli/repo/src/lib/files.ts +15 -0
- package/temp_resend_cli/repo/src/lib/help-text.ts +38 -0
- package/temp_resend_cli/repo/src/lib/output.ts +56 -0
- package/temp_resend_cli/repo/src/lib/pagination.ts +36 -0
- package/temp_resend_cli/repo/src/lib/prompts.ts +149 -0
- package/temp_resend_cli/repo/src/lib/spinner.ts +100 -0
- package/temp_resend_cli/repo/src/lib/table.ts +57 -0
- package/temp_resend_cli/repo/src/lib/tty.ts +28 -0
- package/temp_resend_cli/repo/src/lib/update-check.ts +169 -0
- package/temp_resend_cli/repo/src/lib/version.ts +4 -0
- package/temp_resend_cli/repo/tests/commands/api-keys/create.test.ts +196 -0
- package/temp_resend_cli/repo/tests/commands/api-keys/delete.test.ts +157 -0
- package/temp_resend_cli/repo/tests/commands/api-keys/list.test.ts +134 -0
- package/temp_resend_cli/repo/tests/commands/auth/login.test.ts +153 -0
- package/temp_resend_cli/repo/tests/commands/auth/logout.test.ts +153 -0
- package/temp_resend_cli/repo/tests/commands/broadcasts/create.test.ts +454 -0
- package/temp_resend_cli/repo/tests/commands/broadcasts/delete.test.ts +183 -0
- package/temp_resend_cli/repo/tests/commands/broadcasts/get.test.ts +147 -0
- package/temp_resend_cli/repo/tests/commands/broadcasts/list.test.ts +199 -0
- package/temp_resend_cli/repo/tests/commands/broadcasts/send.test.ts +162 -0
- package/temp_resend_cli/repo/tests/commands/broadcasts/update.test.ts +288 -0
- package/temp_resend_cli/repo/tests/commands/contact-properties/create.test.ts +251 -0
- package/temp_resend_cli/repo/tests/commands/contact-properties/delete.test.ts +184 -0
- package/temp_resend_cli/repo/tests/commands/contact-properties/get.test.ts +145 -0
- package/temp_resend_cli/repo/tests/commands/contact-properties/list.test.ts +181 -0
- package/temp_resend_cli/repo/tests/commands/contact-properties/update.test.ts +217 -0
- package/temp_resend_cli/repo/tests/commands/contacts/add-segment.test.ts +189 -0
- package/temp_resend_cli/repo/tests/commands/contacts/create.test.ts +271 -0
- package/temp_resend_cli/repo/tests/commands/contacts/delete.test.ts +193 -0
- package/temp_resend_cli/repo/tests/commands/contacts/get.test.ts +149 -0
- package/temp_resend_cli/repo/tests/commands/contacts/list.test.ts +176 -0
- package/temp_resend_cli/repo/tests/commands/contacts/remove-segment.test.ts +167 -0
- package/temp_resend_cli/repo/tests/commands/contacts/segments.test.ts +168 -0
- package/temp_resend_cli/repo/tests/commands/contacts/topics.test.ts +164 -0
- package/temp_resend_cli/repo/tests/commands/contacts/update-topics.test.ts +248 -0
- package/temp_resend_cli/repo/tests/commands/contacts/update.test.ts +206 -0
- package/temp_resend_cli/repo/tests/commands/doctor.test.ts +164 -0
- package/temp_resend_cli/repo/tests/commands/domains/create.test.ts +193 -0
- package/temp_resend_cli/repo/tests/commands/domains/delete.test.ts +157 -0
- package/temp_resend_cli/repo/tests/commands/domains/get.test.ts +138 -0
- package/temp_resend_cli/repo/tests/commands/domains/list.test.ts +165 -0
- package/temp_resend_cli/repo/tests/commands/domains/update.test.ts +224 -0
- package/temp_resend_cli/repo/tests/commands/domains/verify.test.ts +118 -0
- package/temp_resend_cli/repo/tests/commands/emails/batch.test.ts +324 -0
- package/temp_resend_cli/repo/tests/commands/emails/get.test.ts +132 -0
- package/temp_resend_cli/repo/tests/commands/emails/receiving/attachment.test.ts +141 -0
- package/temp_resend_cli/repo/tests/commands/emails/receiving/attachments.test.ts +169 -0
- package/temp_resend_cli/repo/tests/commands/emails/receiving/get.test.ts +141 -0
- package/temp_resend_cli/repo/tests/commands/emails/receiving/list.test.ts +182 -0
- package/temp_resend_cli/repo/tests/commands/emails/send.test.ts +312 -0
- package/temp_resend_cli/repo/tests/commands/segments/create.test.ts +164 -0
- package/temp_resend_cli/repo/tests/commands/segments/delete.test.ts +183 -0
- package/temp_resend_cli/repo/tests/commands/segments/get.test.ts +138 -0
- package/temp_resend_cli/repo/tests/commands/segments/list.test.ts +174 -0
- package/temp_resend_cli/repo/tests/commands/teams/list.test.ts +62 -0
- package/temp_resend_cli/repo/tests/commands/teams/remove.test.ts +110 -0
- package/temp_resend_cli/repo/tests/commands/teams/switch.test.ts +103 -0
- package/temp_resend_cli/repo/tests/commands/topics/create.test.ts +192 -0
- package/temp_resend_cli/repo/tests/commands/topics/delete.test.ts +157 -0
- package/temp_resend_cli/repo/tests/commands/topics/get.test.ts +126 -0
- package/temp_resend_cli/repo/tests/commands/topics/list.test.ts +125 -0
- package/temp_resend_cli/repo/tests/commands/topics/update.test.ts +178 -0
- package/temp_resend_cli/repo/tests/commands/webhooks/create.test.ts +225 -0
- package/temp_resend_cli/repo/tests/commands/webhooks/delete.test.ts +157 -0
- package/temp_resend_cli/repo/tests/commands/webhooks/get.test.ts +126 -0
- package/temp_resend_cli/repo/tests/commands/webhooks/list.test.ts +178 -0
- package/temp_resend_cli/repo/tests/commands/webhooks/update.test.ts +207 -0
- package/temp_resend_cli/repo/tests/commands/whoami.test.ts +98 -0
- package/temp_resend_cli/repo/tests/e2e/smoke.test.ts +93 -0
- package/temp_resend_cli/repo/tests/helpers.ts +86 -0
- package/temp_resend_cli/repo/tests/lib/client.test.ts +71 -0
- package/temp_resend_cli/repo/tests/lib/config.test.ts +451 -0
- package/temp_resend_cli/repo/tests/lib/files.test.ts +73 -0
- package/temp_resend_cli/repo/tests/lib/help-text.test.ts +97 -0
- package/temp_resend_cli/repo/tests/lib/output.test.ts +136 -0
- package/temp_resend_cli/repo/tests/lib/prompts.test.ts +185 -0
- package/temp_resend_cli/repo/tests/lib/spinner.test.ts +166 -0
- package/temp_resend_cli/repo/tests/lib/table.test.ts +63 -0
- package/temp_resend_cli/repo/tests/lib/tty.test.ts +89 -0
- package/temp_resend_cli/repo/tests/lib/update-check.test.ts +179 -0
- package/temp_resend_cli/repo/tsconfig.json +14 -0
- package/temp_resend_cli/repo/vitest.config.e2e.ts +8 -0
- package/temp_resend_cli/repo/vitest.config.ts +10 -0
- package/tests/test-mcp-browser-use-smoke.sh +28 -56
- package/tests/test-monty-smoke.sh +32 -0
- package/tests/test-resend-smoke.sh +36 -0
|
@@ -0,0 +1,1373 @@
|
|
|
1
|
+
use std::fmt::Write;
|
|
2
|
+
|
|
3
|
+
use ahash::AHashSet;
|
|
4
|
+
use hashbrown::HashTable;
|
|
5
|
+
|
|
6
|
+
use super::{MontyIter, PyTrait};
|
|
7
|
+
use crate::{
|
|
8
|
+
args::ArgValues,
|
|
9
|
+
bytecode::{CallResult, VM},
|
|
10
|
+
defer_drop, defer_drop_mut,
|
|
11
|
+
exception_private::{ExcType, RunResult},
|
|
12
|
+
heap::{ContainsHeap, DropWithHeap, Heap, HeapData, HeapGuard, HeapId},
|
|
13
|
+
intern::{Interns, StaticStrings},
|
|
14
|
+
resource::{ResourceError, ResourceTracker},
|
|
15
|
+
types::Type,
|
|
16
|
+
value::{EitherStr, Value},
|
|
17
|
+
};
|
|
18
|
+
|
|
19
|
+
/// Entry in the set storage, containing a value and its cached hash.
|
|
20
|
+
#[derive(Debug, serde::Serialize, serde::Deserialize)]
|
|
21
|
+
struct SetEntry {
|
|
22
|
+
pub(crate) value: Value,
|
|
23
|
+
/// Cached hash for efficient lookup and reinsertion.
|
|
24
|
+
pub(crate) hash: u64,
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
/// Internal storage shared between Set and FrozenSet.
|
|
28
|
+
///
|
|
29
|
+
/// Uses a `HashTable<usize>` for O(1) lookups combined with a dense `Vec<SetEntry>`
|
|
30
|
+
/// to preserve insertion order (consistent with Python 3.7+ dict behavior).
|
|
31
|
+
/// The hash table maps value hashes to indices in the entries vector.
|
|
32
|
+
#[derive(Debug, Default)]
|
|
33
|
+
pub(crate) struct SetStorage {
|
|
34
|
+
/// Maps hash to index in entries vector.
|
|
35
|
+
indices: HashTable<usize>,
|
|
36
|
+
/// Dense vector of entries maintaining insertion order.
|
|
37
|
+
entries: Vec<SetEntry>,
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
impl SetStorage {
|
|
41
|
+
/// Creates a new empty set storage.
|
|
42
|
+
fn new() -> Self {
|
|
43
|
+
Self::default()
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
/// Creates a new set storage with pre-allocated capacity.
|
|
47
|
+
fn with_capacity(capacity: usize) -> Self {
|
|
48
|
+
Self {
|
|
49
|
+
indices: HashTable::with_capacity(capacity),
|
|
50
|
+
entries: Vec::with_capacity(capacity),
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
/// Creates a SetStorage from a vector of (value, hash) pairs.
|
|
55
|
+
///
|
|
56
|
+
/// This is used to avoid borrow conflicts when we need to copy another set's
|
|
57
|
+
/// contents and then perform operations requiring mutable heap access.
|
|
58
|
+
/// The caller is responsible for handling reference counting.
|
|
59
|
+
fn from_entries(entries: Vec<(Value, u64)>) -> Self {
|
|
60
|
+
let mut storage = Self::with_capacity(entries.len());
|
|
61
|
+
for (idx, (value, hash)) in entries.into_iter().enumerate() {
|
|
62
|
+
storage.entries.push(SetEntry { value, hash });
|
|
63
|
+
storage.indices.insert_unique(hash, idx, |&i| storage.entries[i].hash);
|
|
64
|
+
}
|
|
65
|
+
storage
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
/// Clones entries with proper reference counting.
|
|
69
|
+
fn clone_entries(&self, heap: &impl ContainsHeap) -> Vec<(Value, u64)> {
|
|
70
|
+
self.entries
|
|
71
|
+
.iter()
|
|
72
|
+
.map(|e| (e.value.clone_with_heap(heap), e.hash))
|
|
73
|
+
.collect()
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
/// Returns the number of elements in the set.
|
|
77
|
+
fn len(&self) -> usize {
|
|
78
|
+
self.entries.len()
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
/// Returns true if the set is empty.
|
|
82
|
+
fn is_empty(&self) -> bool {
|
|
83
|
+
self.entries.is_empty()
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
/// Returns whether this set contains any heap references (`Value::Ref`).
|
|
87
|
+
///
|
|
88
|
+
/// Used during allocation to determine if this container could create cycles.
|
|
89
|
+
#[inline]
|
|
90
|
+
fn has_refs(&self) -> bool {
|
|
91
|
+
self.entries.iter().any(|e| matches!(e.value, Value::Ref(_)))
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
/// Adds an element to the set, transferring ownership.
|
|
95
|
+
///
|
|
96
|
+
/// Returns `Ok(true)` if the element was added (not already present),
|
|
97
|
+
/// `Ok(false)` if the element was already in the set.
|
|
98
|
+
/// Returns `Err` if the element is unhashable.
|
|
99
|
+
///
|
|
100
|
+
/// The caller transfers ownership of `value`. If the value is already in
|
|
101
|
+
/// the set, it will be dropped.
|
|
102
|
+
fn add(&mut self, value: Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<bool> {
|
|
103
|
+
let hash = match value.py_hash(vm.heap, vm.interns) {
|
|
104
|
+
Ok(Some(h)) => h,
|
|
105
|
+
Ok(None) => {
|
|
106
|
+
let err = ExcType::type_error_unhashable_set_element(value.py_type(vm.heap));
|
|
107
|
+
value.drop_with_heap(vm.heap);
|
|
108
|
+
return Err(err);
|
|
109
|
+
}
|
|
110
|
+
Err(e) => {
|
|
111
|
+
value.drop_with_heap(vm.heap);
|
|
112
|
+
return Err(e.into());
|
|
113
|
+
}
|
|
114
|
+
};
|
|
115
|
+
|
|
116
|
+
// Check if value already exists.
|
|
117
|
+
let existing = self
|
|
118
|
+
.indices
|
|
119
|
+
.find(hash, |&idx| value.py_eq(&self.entries[idx].value, vm).unwrap_or(false));
|
|
120
|
+
|
|
121
|
+
if existing.is_some() {
|
|
122
|
+
// Value already in set, drop the new value
|
|
123
|
+
value.drop_with_heap(vm.heap);
|
|
124
|
+
Ok(false)
|
|
125
|
+
} else {
|
|
126
|
+
// Add new entry
|
|
127
|
+
let index = self.entries.len();
|
|
128
|
+
self.entries.push(SetEntry { value, hash });
|
|
129
|
+
self.indices.insert_unique(hash, index, |&idx| self.entries[idx].hash);
|
|
130
|
+
Ok(true)
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
/// Removes an element from the set.
|
|
135
|
+
///
|
|
136
|
+
/// Returns `Ok(true)` if the element was removed, `Ok(false)` if not found.
|
|
137
|
+
/// Returns `Err` if the key is unhashable.
|
|
138
|
+
fn remove(&mut self, value: &Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<bool> {
|
|
139
|
+
let hash = value
|
|
140
|
+
.py_hash(vm.heap, vm.interns)?
|
|
141
|
+
.ok_or_else(|| ExcType::type_error_unhashable_set_element(value.py_type(vm.heap)))?;
|
|
142
|
+
|
|
143
|
+
let entry = self.indices.entry(
|
|
144
|
+
hash,
|
|
145
|
+
|&idx| value.py_eq(&self.entries[idx].value, vm).unwrap_or(false),
|
|
146
|
+
|&idx| self.entries[idx].hash,
|
|
147
|
+
);
|
|
148
|
+
|
|
149
|
+
if let hashbrown::hash_table::Entry::Occupied(occ) = entry {
|
|
150
|
+
let index = *occ.get();
|
|
151
|
+
let removed_entry = self.entries.remove(index);
|
|
152
|
+
occ.remove();
|
|
153
|
+
|
|
154
|
+
// Update indices for entries that shifted down
|
|
155
|
+
for idx in &mut self.indices {
|
|
156
|
+
if *idx > index {
|
|
157
|
+
*idx -= 1;
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
// Drop the removed value
|
|
162
|
+
removed_entry.value.drop_with_heap(vm);
|
|
163
|
+
Ok(true)
|
|
164
|
+
} else {
|
|
165
|
+
Ok(false)
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
/// Removes an element from the set without raising an error if not found.
|
|
170
|
+
///
|
|
171
|
+
/// Returns `Ok(())` always (unless the key is unhashable).
|
|
172
|
+
fn discard(&mut self, value: &Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<()> {
|
|
173
|
+
self.remove(value, vm)?;
|
|
174
|
+
Ok(())
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
/// Removes and returns an arbitrary element from the set.
|
|
178
|
+
///
|
|
179
|
+
/// Returns `Err(KeyError)` if the set is empty.
|
|
180
|
+
fn pop(&mut self) -> RunResult<Value> {
|
|
181
|
+
if self.entries.is_empty() {
|
|
182
|
+
return Err(ExcType::key_error_pop_empty_set());
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
// Remove the last entry (most efficient)
|
|
186
|
+
let entry = self.entries.pop().expect("checked non-empty");
|
|
187
|
+
|
|
188
|
+
// Remove from hash table
|
|
189
|
+
self.indices
|
|
190
|
+
.find_entry(entry.hash, |&idx| idx == self.entries.len())
|
|
191
|
+
.expect("entry must exist")
|
|
192
|
+
.remove();
|
|
193
|
+
|
|
194
|
+
Ok(entry.value)
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
/// Removes all elements from the set.
|
|
198
|
+
fn clear(&mut self, heap: &mut Heap<impl ResourceTracker>) {
|
|
199
|
+
self.entries.drain(..).drop_with_heap(heap);
|
|
200
|
+
self.indices.clear();
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
/// Creates a deep clone with proper reference counting.
|
|
204
|
+
fn clone_with_heap(&self, heap: &impl ContainsHeap) -> Self {
|
|
205
|
+
Self {
|
|
206
|
+
indices: self.indices.clone(),
|
|
207
|
+
entries: self
|
|
208
|
+
.entries
|
|
209
|
+
.iter()
|
|
210
|
+
.map(|entry| SetEntry {
|
|
211
|
+
value: entry.value.clone_with_heap(heap),
|
|
212
|
+
hash: entry.hash,
|
|
213
|
+
})
|
|
214
|
+
.collect(),
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
/// Checks if the set contains a value.
|
|
219
|
+
pub fn contains(&self, value: &Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<bool> {
|
|
220
|
+
let hash = value
|
|
221
|
+
.py_hash(vm.heap, vm.interns)?
|
|
222
|
+
.ok_or_else(|| ExcType::type_error_unhashable_set_element(value.py_type(vm.heap)))?;
|
|
223
|
+
|
|
224
|
+
// Set values are typically shallow (strings, ints, tuples of primitives),
|
|
225
|
+
// so recursion errors are unlikely. If one occurs, treat it as "not equal".
|
|
226
|
+
Ok(self
|
|
227
|
+
.indices
|
|
228
|
+
.find(hash, |&idx| value.py_eq(&self.entries[idx].value, vm).unwrap_or(false))
|
|
229
|
+
.is_some())
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
/// Returns an iterator over the values in the set.
|
|
233
|
+
pub(crate) fn iter(&self) -> impl Iterator<Item = &Value> {
|
|
234
|
+
self.entries.iter().map(|e| &e.value)
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
/// Returns the value at the given index, if valid.
|
|
238
|
+
///
|
|
239
|
+
/// Used by MontyIter for index-based iteration.
|
|
240
|
+
pub(crate) fn value_at(&self, index: usize) -> Option<&Value> {
|
|
241
|
+
self.entries.get(index).map(|e| &e.value)
|
|
242
|
+
}
|
|
243
|
+
|
|
244
|
+
/// Collects heap IDs for reference counting cleanup.
|
|
245
|
+
fn collect_dec_ref_ids(&mut self, stack: &mut Vec<HeapId>) {
|
|
246
|
+
for entry in &mut self.entries {
|
|
247
|
+
if let Value::Ref(id) = &entry.value {
|
|
248
|
+
stack.push(*id);
|
|
249
|
+
#[cfg(feature = "ref-count-panic")]
|
|
250
|
+
entry.value.dec_ref_forget();
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
|
|
255
|
+
/// Compares two sets for equality.
|
|
256
|
+
fn eq(&self, other: &Self, vm: &mut VM<'_, '_, impl ResourceTracker>) -> Result<bool, ResourceError> {
|
|
257
|
+
if self.len() != other.len() {
|
|
258
|
+
return Ok(false);
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
let token = vm.heap.incr_recursion_depth()?;
|
|
262
|
+
defer_drop!(token, vm);
|
|
263
|
+
// Check that every element in self is in other
|
|
264
|
+
for entry in &self.entries {
|
|
265
|
+
if !matches!(other.contains(&entry.value, vm), Ok(true)) {
|
|
266
|
+
return Ok(false);
|
|
267
|
+
}
|
|
268
|
+
}
|
|
269
|
+
Ok(true)
|
|
270
|
+
}
|
|
271
|
+
|
|
272
|
+
/// Returns true if this set is a subset of other.
|
|
273
|
+
fn is_subset(&self, other: &Self, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<bool> {
|
|
274
|
+
for entry in &self.entries {
|
|
275
|
+
if !other.contains(&entry.value, vm)? {
|
|
276
|
+
return Ok(false);
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
Ok(true)
|
|
280
|
+
}
|
|
281
|
+
|
|
282
|
+
/// Returns true if this set is a superset of other.
|
|
283
|
+
fn is_superset(&self, other: &Self, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<bool> {
|
|
284
|
+
other.is_subset(self, vm)
|
|
285
|
+
}
|
|
286
|
+
|
|
287
|
+
/// Returns true if this set has no elements in common with other.
|
|
288
|
+
fn is_disjoint(&self, other: &Self, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<bool> {
|
|
289
|
+
// Iterate over the smaller set for efficiency
|
|
290
|
+
let (smaller, larger) = if self.len() <= other.len() {
|
|
291
|
+
(self, other)
|
|
292
|
+
} else {
|
|
293
|
+
(other, self)
|
|
294
|
+
};
|
|
295
|
+
|
|
296
|
+
for entry in &smaller.entries {
|
|
297
|
+
if larger.contains(&entry.value, vm)? {
|
|
298
|
+
return Ok(false);
|
|
299
|
+
}
|
|
300
|
+
}
|
|
301
|
+
Ok(true)
|
|
302
|
+
}
|
|
303
|
+
|
|
304
|
+
/// Returns a new set containing elements in either set (union).
|
|
305
|
+
fn union(&self, other: &Self, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<Self> {
|
|
306
|
+
let mut result_guard = HeapGuard::new(self.clone_with_heap(vm), vm);
|
|
307
|
+
let (result, vm) = result_guard.as_parts_mut();
|
|
308
|
+
for entry in &other.entries {
|
|
309
|
+
let value = entry.value.clone_with_heap(vm);
|
|
310
|
+
result.add(value, vm)?;
|
|
311
|
+
}
|
|
312
|
+
Ok(result_guard.into_inner())
|
|
313
|
+
}
|
|
314
|
+
|
|
315
|
+
/// Returns a new set containing elements in both sets (intersection).
|
|
316
|
+
fn intersection(&self, other: &Self, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<Self> {
|
|
317
|
+
let mut result_guard = HeapGuard::new(Self::new(), vm);
|
|
318
|
+
let (result, vm) = result_guard.as_parts_mut();
|
|
319
|
+
// Iterate over the smaller set for efficiency
|
|
320
|
+
let (smaller, larger) = if self.len() <= other.len() {
|
|
321
|
+
(self, other)
|
|
322
|
+
} else {
|
|
323
|
+
(other, self)
|
|
324
|
+
};
|
|
325
|
+
|
|
326
|
+
for entry in &smaller.entries {
|
|
327
|
+
if larger.contains(&entry.value, vm)? {
|
|
328
|
+
let value = entry.value.clone_with_heap(vm);
|
|
329
|
+
result.add(value, vm)?;
|
|
330
|
+
}
|
|
331
|
+
}
|
|
332
|
+
Ok(result_guard.into_inner())
|
|
333
|
+
}
|
|
334
|
+
|
|
335
|
+
/// Returns a new set containing elements in self but not in other (difference).
|
|
336
|
+
fn difference(&self, other: &Self, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<Self> {
|
|
337
|
+
let mut result_guard = HeapGuard::new(Self::new(), vm);
|
|
338
|
+
let (result, vm) = result_guard.as_parts_mut();
|
|
339
|
+
for entry in &self.entries {
|
|
340
|
+
if !other.contains(&entry.value, vm)? {
|
|
341
|
+
let value = entry.value.clone_with_heap(vm);
|
|
342
|
+
result.add(value, vm)?;
|
|
343
|
+
}
|
|
344
|
+
}
|
|
345
|
+
Ok(result_guard.into_inner())
|
|
346
|
+
}
|
|
347
|
+
|
|
348
|
+
/// Returns a new set containing elements in either set but not both (symmetric difference).
|
|
349
|
+
fn symmetric_difference(&self, other: &Self, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<Self> {
|
|
350
|
+
let mut result_guard = HeapGuard::new(Self::new(), vm);
|
|
351
|
+
let (result, vm) = result_guard.as_parts_mut();
|
|
352
|
+
|
|
353
|
+
// Add elements in self but not in other
|
|
354
|
+
for entry in &self.entries {
|
|
355
|
+
if !other.contains(&entry.value, vm)? {
|
|
356
|
+
let value = entry.value.clone_with_heap(vm);
|
|
357
|
+
result.add(value, vm)?;
|
|
358
|
+
}
|
|
359
|
+
}
|
|
360
|
+
|
|
361
|
+
// Add elements in other but not in self
|
|
362
|
+
for entry in &other.entries {
|
|
363
|
+
if !self.contains(&entry.value, vm)? {
|
|
364
|
+
let value = entry.value.clone_with_heap(vm);
|
|
365
|
+
result.add(value, vm)?;
|
|
366
|
+
}
|
|
367
|
+
}
|
|
368
|
+
|
|
369
|
+
Ok(result_guard.into_inner())
|
|
370
|
+
}
|
|
371
|
+
|
|
372
|
+
/// Adds all elements from other to this set (in-place union).
|
|
373
|
+
fn update(&mut self, other: &Self, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<()> {
|
|
374
|
+
for entry in &other.entries {
|
|
375
|
+
let value = entry.value.clone_with_heap(vm);
|
|
376
|
+
self.add(value, vm)?;
|
|
377
|
+
}
|
|
378
|
+
Ok(())
|
|
379
|
+
}
|
|
380
|
+
|
|
381
|
+
/// Writes the repr format to a formatter.
|
|
382
|
+
///
|
|
383
|
+
/// For sets, outputs `{elem1, elem2, ...}` (no type prefix).
|
|
384
|
+
/// For frozensets, outputs `frozenset({elem1, elem2, ...})`.
|
|
385
|
+
fn repr_fmt(
|
|
386
|
+
&self,
|
|
387
|
+
f: &mut impl Write,
|
|
388
|
+
vm: &VM<'_, '_, impl ResourceTracker>,
|
|
389
|
+
heap_ids: &mut AHashSet<HeapId>,
|
|
390
|
+
type_name: &str,
|
|
391
|
+
) -> std::fmt::Result {
|
|
392
|
+
if self.is_empty() {
|
|
393
|
+
return write!(f, "{type_name}()");
|
|
394
|
+
}
|
|
395
|
+
|
|
396
|
+
// Check depth limit before recursing
|
|
397
|
+
let Some(token) = vm.heap.incr_recursion_depth_for_repr() else {
|
|
398
|
+
return f.write_str("{...}");
|
|
399
|
+
};
|
|
400
|
+
crate::defer_drop_immutable_heap!(token, vm);
|
|
401
|
+
|
|
402
|
+
// frozenset needs type prefix: frozenset({...}), but set doesn't: {...}
|
|
403
|
+
let needs_prefix = type_name != "set";
|
|
404
|
+
if needs_prefix {
|
|
405
|
+
write!(f, "{type_name}(")?;
|
|
406
|
+
}
|
|
407
|
+
|
|
408
|
+
f.write_char('{')?;
|
|
409
|
+
let mut first = true;
|
|
410
|
+
for entry in &self.entries {
|
|
411
|
+
if !first {
|
|
412
|
+
if vm.heap.check_time().is_err() {
|
|
413
|
+
f.write_str(", ...[timeout]")?;
|
|
414
|
+
break;
|
|
415
|
+
}
|
|
416
|
+
f.write_str(", ")?;
|
|
417
|
+
}
|
|
418
|
+
first = false;
|
|
419
|
+
entry.value.py_repr_fmt(f, vm, heap_ids)?;
|
|
420
|
+
}
|
|
421
|
+
f.write_char('}')?;
|
|
422
|
+
|
|
423
|
+
if needs_prefix {
|
|
424
|
+
f.write_char(')')?;
|
|
425
|
+
}
|
|
426
|
+
|
|
427
|
+
Ok(())
|
|
428
|
+
}
|
|
429
|
+
|
|
430
|
+
/// Estimates the memory size of this storage.
|
|
431
|
+
fn estimate_size(&self) -> usize {
|
|
432
|
+
std::mem::size_of::<Self>() + self.len() * std::mem::size_of::<SetEntry>()
|
|
433
|
+
}
|
|
434
|
+
}
|
|
435
|
+
|
|
436
|
+
/// Python set type - mutable, unordered collection of unique hashable elements.
|
|
437
|
+
///
|
|
438
|
+
/// Sets support standard operations like add, remove, discard, pop, clear, as well
|
|
439
|
+
/// as set algebra operations like union, intersection, difference, and symmetric
|
|
440
|
+
/// difference.
|
|
441
|
+
///
|
|
442
|
+
/// # Reference Counting
|
|
443
|
+
/// When values are added, their reference counts are NOT incremented by the set -
|
|
444
|
+
/// the caller transfers ownership. When values are removed or the set is cleared,
|
|
445
|
+
/// their reference counts are decremented.
|
|
446
|
+
#[derive(Debug, Default)]
|
|
447
|
+
pub(crate) struct Set(SetStorage);
|
|
448
|
+
|
|
449
|
+
impl Set {
|
|
450
|
+
/// Creates a new empty set.
|
|
451
|
+
#[must_use]
|
|
452
|
+
pub fn new() -> Self {
|
|
453
|
+
Self(SetStorage::new())
|
|
454
|
+
}
|
|
455
|
+
|
|
456
|
+
/// Creates a set with pre-allocated capacity.
|
|
457
|
+
#[must_use]
|
|
458
|
+
pub fn with_capacity(capacity: usize) -> Self {
|
|
459
|
+
Self(SetStorage::with_capacity(capacity))
|
|
460
|
+
}
|
|
461
|
+
|
|
462
|
+
/// Returns the number of elements in the set.
|
|
463
|
+
#[must_use]
|
|
464
|
+
pub fn len(&self) -> usize {
|
|
465
|
+
self.0.len()
|
|
466
|
+
}
|
|
467
|
+
|
|
468
|
+
/// Returns true if the set is empty.
|
|
469
|
+
#[must_use]
|
|
470
|
+
pub fn is_empty(&self) -> bool {
|
|
471
|
+
self.0.is_empty()
|
|
472
|
+
}
|
|
473
|
+
|
|
474
|
+
/// Returns whether this set contains any heap references (`Value::Ref`).
|
|
475
|
+
///
|
|
476
|
+
/// Used during allocation to determine if this container could create cycles.
|
|
477
|
+
#[inline]
|
|
478
|
+
#[must_use]
|
|
479
|
+
pub fn has_refs(&self) -> bool {
|
|
480
|
+
self.0.has_refs()
|
|
481
|
+
}
|
|
482
|
+
|
|
483
|
+
/// Adds an element to the set, transferring ownership.
|
|
484
|
+
///
|
|
485
|
+
/// Returns `Ok(true)` if added, `Ok(false)` if already present.
|
|
486
|
+
pub fn add(&mut self, value: Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<bool> {
|
|
487
|
+
self.0.add(value, vm)
|
|
488
|
+
}
|
|
489
|
+
|
|
490
|
+
/// Removes an element from the set.
|
|
491
|
+
///
|
|
492
|
+
/// Returns `Err(KeyError)` if the element is not present.
|
|
493
|
+
pub fn remove(&mut self, value: &Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<()> {
|
|
494
|
+
if self.0.remove(value, vm)? {
|
|
495
|
+
Ok(())
|
|
496
|
+
} else {
|
|
497
|
+
Err(ExcType::key_error(value, vm))
|
|
498
|
+
}
|
|
499
|
+
}
|
|
500
|
+
|
|
501
|
+
/// Removes an element from the set if present.
|
|
502
|
+
///
|
|
503
|
+
/// Does not raise an error if the element is not found.
|
|
504
|
+
pub fn discard(&mut self, value: &Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<()> {
|
|
505
|
+
self.0.discard(value, vm)
|
|
506
|
+
}
|
|
507
|
+
|
|
508
|
+
/// Removes and returns an arbitrary element from the set.
|
|
509
|
+
///
|
|
510
|
+
/// Returns `Err(KeyError)` if the set is empty.
|
|
511
|
+
pub fn pop(&mut self) -> RunResult<Value> {
|
|
512
|
+
self.0.pop()
|
|
513
|
+
}
|
|
514
|
+
|
|
515
|
+
/// Removes all elements from the set.
|
|
516
|
+
pub fn clear(&mut self, heap: &mut Heap<impl ResourceTracker>) {
|
|
517
|
+
self.0.clear(heap);
|
|
518
|
+
}
|
|
519
|
+
|
|
520
|
+
/// Returns a shallow copy of the set.
|
|
521
|
+
#[must_use]
|
|
522
|
+
pub fn copy(&self, heap: &mut Heap<impl ResourceTracker>) -> Self {
|
|
523
|
+
Self(self.0.clone_with_heap(heap))
|
|
524
|
+
}
|
|
525
|
+
|
|
526
|
+
/// Checks if the set contains a value.
|
|
527
|
+
pub fn contains(&self, value: &Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<bool> {
|
|
528
|
+
self.0.contains(value, vm)
|
|
529
|
+
}
|
|
530
|
+
|
|
531
|
+
/// Returns the internal storage (for set operations between Set and FrozenSet).
|
|
532
|
+
pub(crate) fn storage(&self) -> &SetStorage {
|
|
533
|
+
&self.0
|
|
534
|
+
}
|
|
535
|
+
|
|
536
|
+
/// Returns an iterator over the set's elements in insertion order.
|
|
537
|
+
///
|
|
538
|
+
/// This is primarily used by other runtime helpers that need to implement
|
|
539
|
+
/// set-like protocols while still preserving Monty's single canonical set
|
|
540
|
+
/// storage implementation.
|
|
541
|
+
pub(crate) fn iter(&self) -> impl Iterator<Item = &Value> {
|
|
542
|
+
self.0.iter()
|
|
543
|
+
}
|
|
544
|
+
|
|
545
|
+
/// Creates a set from the `set()` constructor call.
|
|
546
|
+
///
|
|
547
|
+
/// - `set()` with no args returns an empty set
|
|
548
|
+
/// - `set(iterable)` creates a set from any iterable (list, tuple, set, dict, range, str, bytes)
|
|
549
|
+
pub fn init(vm: &mut VM<'_, '_, impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
|
|
550
|
+
let value = args.get_zero_one_arg("set", vm.heap)?;
|
|
551
|
+
let set = match value {
|
|
552
|
+
None => Self::new(),
|
|
553
|
+
Some(v) => Self::from_iterable(v, vm)?,
|
|
554
|
+
};
|
|
555
|
+
let heap_id = vm.heap.allocate(HeapData::Set(set))?;
|
|
556
|
+
Ok(Value::Ref(heap_id))
|
|
557
|
+
}
|
|
558
|
+
|
|
559
|
+
/// Creates a set from a MontyIter, adding elements one by one.
|
|
560
|
+
///
|
|
561
|
+
/// Unlike list/tuple which can just collect into a Vec, sets need to add
|
|
562
|
+
/// each element individually to handle duplicates and compute hashes.
|
|
563
|
+
fn from_iterator(iter: MontyIter, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<Self> {
|
|
564
|
+
defer_drop_mut!(iter, vm);
|
|
565
|
+
let mut set = Self::with_capacity(iter.size_hint(vm.heap));
|
|
566
|
+
while let Some(item) = iter.for_next(vm)? {
|
|
567
|
+
set.add(item, vm)?;
|
|
568
|
+
}
|
|
569
|
+
Ok(set)
|
|
570
|
+
}
|
|
571
|
+
|
|
572
|
+
/// Creates a set from an iterable value.
|
|
573
|
+
///
|
|
574
|
+
/// This is a convenience method used by helper methods that need to convert
|
|
575
|
+
/// arbitrary iterables to sets. It uses `MontyIter` internally.
|
|
576
|
+
fn from_iterable(iterable: Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<Self> {
|
|
577
|
+
let iter = MontyIter::new(iterable, vm)?;
|
|
578
|
+
let set = Self::from_iterator(iter, vm)?;
|
|
579
|
+
Ok(set)
|
|
580
|
+
}
|
|
581
|
+
}
|
|
582
|
+
|
|
583
|
+
impl DropWithHeap for Set {
|
|
584
|
+
fn drop_with_heap<H: ContainsHeap>(self, heap: &mut H) {
|
|
585
|
+
self.0.drop_with_heap(heap);
|
|
586
|
+
}
|
|
587
|
+
}
|
|
588
|
+
|
|
589
|
+
impl DropWithHeap for SetStorage {
|
|
590
|
+
fn drop_with_heap<H: ContainsHeap>(self, heap: &mut H) {
|
|
591
|
+
self.entries.drop_with_heap(heap);
|
|
592
|
+
}
|
|
593
|
+
}
|
|
594
|
+
|
|
595
|
+
impl DropWithHeap for FrozenSet {
|
|
596
|
+
fn drop_with_heap<H: ContainsHeap>(self, heap: &mut H) {
|
|
597
|
+
self.0.drop_with_heap(heap);
|
|
598
|
+
}
|
|
599
|
+
}
|
|
600
|
+
|
|
601
|
+
impl DropWithHeap for SetEntry {
|
|
602
|
+
fn drop_with_heap<H: ContainsHeap>(self, heap: &mut H) {
|
|
603
|
+
self.value.drop_with_heap(heap);
|
|
604
|
+
}
|
|
605
|
+
}
|
|
606
|
+
|
|
607
|
+
impl PyTrait for Set {
|
|
608
|
+
fn py_type(&self, _heap: &Heap<impl ResourceTracker>) -> Type {
|
|
609
|
+
Type::Set
|
|
610
|
+
}
|
|
611
|
+
|
|
612
|
+
fn py_estimate_size(&self) -> usize {
|
|
613
|
+
self.0.estimate_size()
|
|
614
|
+
}
|
|
615
|
+
|
|
616
|
+
fn py_len(&self, _vm: &VM<'_, '_, impl ResourceTracker>) -> Option<usize> {
|
|
617
|
+
Some(self.len())
|
|
618
|
+
}
|
|
619
|
+
|
|
620
|
+
fn py_eq(&self, other: &Self, vm: &mut VM<'_, '_, impl ResourceTracker>) -> Result<bool, ResourceError> {
|
|
621
|
+
if self.len() != other.len() {
|
|
622
|
+
return Ok(false);
|
|
623
|
+
}
|
|
624
|
+
let token = vm.heap.incr_recursion_depth()?;
|
|
625
|
+
defer_drop!(token, vm);
|
|
626
|
+
self.0.eq(&other.0, vm)
|
|
627
|
+
}
|
|
628
|
+
|
|
629
|
+
fn py_dec_ref_ids(&mut self, stack: &mut Vec<HeapId>) {
|
|
630
|
+
self.0.collect_dec_ref_ids(stack);
|
|
631
|
+
}
|
|
632
|
+
|
|
633
|
+
fn py_bool(&self, _vm: &VM<'_, '_, impl ResourceTracker>) -> bool {
|
|
634
|
+
!self.is_empty()
|
|
635
|
+
}
|
|
636
|
+
|
|
637
|
+
fn py_repr_fmt(
|
|
638
|
+
&self,
|
|
639
|
+
f: &mut impl Write,
|
|
640
|
+
vm: &VM<'_, '_, impl ResourceTracker>,
|
|
641
|
+
heap_ids: &mut AHashSet<HeapId>,
|
|
642
|
+
) -> std::fmt::Result {
|
|
643
|
+
self.0.repr_fmt(f, vm, heap_ids, "set")
|
|
644
|
+
}
|
|
645
|
+
|
|
646
|
+
fn py_call_attr(
|
|
647
|
+
&mut self,
|
|
648
|
+
_self_id: HeapId,
|
|
649
|
+
vm: &mut VM<'_, '_, impl ResourceTracker>,
|
|
650
|
+
attr: &EitherStr,
|
|
651
|
+
args: ArgValues,
|
|
652
|
+
) -> RunResult<CallResult> {
|
|
653
|
+
let value = match attr.static_string() {
|
|
654
|
+
Some(StaticStrings::Add) => {
|
|
655
|
+
let value = args.get_one_arg("set.add", vm.heap)?;
|
|
656
|
+
self.add(value, vm)?;
|
|
657
|
+
Ok(Value::None)
|
|
658
|
+
}
|
|
659
|
+
Some(StaticStrings::Remove) => {
|
|
660
|
+
let value = args.get_one_arg("set.remove", vm.heap)?;
|
|
661
|
+
defer_drop!(value, vm);
|
|
662
|
+
self.remove(value, vm)?;
|
|
663
|
+
Ok(Value::None)
|
|
664
|
+
}
|
|
665
|
+
Some(StaticStrings::Discard) => {
|
|
666
|
+
let value = args.get_one_arg("set.discard", vm.heap)?;
|
|
667
|
+
defer_drop!(value, vm);
|
|
668
|
+
self.discard(value, vm)?;
|
|
669
|
+
Ok(Value::None)
|
|
670
|
+
}
|
|
671
|
+
Some(StaticStrings::Pop) => {
|
|
672
|
+
args.check_zero_args("set.pop", vm.heap)?;
|
|
673
|
+
self.pop()
|
|
674
|
+
}
|
|
675
|
+
Some(StaticStrings::Clear) => {
|
|
676
|
+
args.check_zero_args("set.clear", vm.heap)?;
|
|
677
|
+
self.clear(vm.heap);
|
|
678
|
+
Ok(Value::None)
|
|
679
|
+
}
|
|
680
|
+
Some(StaticStrings::Copy) => {
|
|
681
|
+
args.check_zero_args("set.copy", vm.heap)?;
|
|
682
|
+
let copy = self.copy(vm.heap);
|
|
683
|
+
let heap_id = vm.heap.allocate(HeapData::Set(copy))?;
|
|
684
|
+
Ok(Value::Ref(heap_id))
|
|
685
|
+
}
|
|
686
|
+
Some(StaticStrings::Update) => {
|
|
687
|
+
let other = args.get_one_arg("set.update", vm.heap)?;
|
|
688
|
+
self.update_from_value(other, vm)?;
|
|
689
|
+
Ok(Value::None)
|
|
690
|
+
}
|
|
691
|
+
Some(StaticStrings::Union) => {
|
|
692
|
+
let other = args.get_one_arg("set.union", vm.heap)?;
|
|
693
|
+
let result = self.union_from_value(other, vm)?;
|
|
694
|
+
let heap_id = vm.heap.allocate(HeapData::Set(result))?;
|
|
695
|
+
Ok(Value::Ref(heap_id))
|
|
696
|
+
}
|
|
697
|
+
Some(StaticStrings::Intersection) => {
|
|
698
|
+
let other = args.get_one_arg("set.intersection", vm.heap)?;
|
|
699
|
+
let result = self.intersection_from_value(other, vm)?;
|
|
700
|
+
let heap_id = vm.heap.allocate(HeapData::Set(result))?;
|
|
701
|
+
Ok(Value::Ref(heap_id))
|
|
702
|
+
}
|
|
703
|
+
Some(StaticStrings::Difference) => {
|
|
704
|
+
let other = args.get_one_arg("set.difference", vm.heap)?;
|
|
705
|
+
let result = self.difference_from_value(other, vm)?;
|
|
706
|
+
let heap_id = vm.heap.allocate(HeapData::Set(result))?;
|
|
707
|
+
Ok(Value::Ref(heap_id))
|
|
708
|
+
}
|
|
709
|
+
Some(StaticStrings::SymmetricDifference) => {
|
|
710
|
+
let other = args.get_one_arg("set.symmetric_difference", vm.heap)?;
|
|
711
|
+
let result = self.symmetric_difference_from_value(other, vm)?;
|
|
712
|
+
let heap_id = vm.heap.allocate(HeapData::Set(result))?;
|
|
713
|
+
Ok(Value::Ref(heap_id))
|
|
714
|
+
}
|
|
715
|
+
Some(StaticStrings::Issubset) => {
|
|
716
|
+
let other = args.get_one_arg("set.issubset", vm.heap)?;
|
|
717
|
+
defer_drop!(other, vm);
|
|
718
|
+
Ok(Value::Bool(self.issubset_from_value(other, vm)?))
|
|
719
|
+
}
|
|
720
|
+
Some(StaticStrings::Issuperset) => {
|
|
721
|
+
let other = args.get_one_arg("set.issuperset", vm.heap)?;
|
|
722
|
+
defer_drop!(other, vm);
|
|
723
|
+
Ok(Value::Bool(self.issuperset_from_value(other, vm)?))
|
|
724
|
+
}
|
|
725
|
+
Some(StaticStrings::Isdisjoint) => {
|
|
726
|
+
let other = args.get_one_arg("set.isdisjoint", vm.heap)?;
|
|
727
|
+
defer_drop!(other, vm);
|
|
728
|
+
Ok(Value::Bool(self.isdisjoint_from_value(other, vm)?))
|
|
729
|
+
}
|
|
730
|
+
_ => {
|
|
731
|
+
args.drop_with_heap(vm);
|
|
732
|
+
return Err(ExcType::attribute_error(Type::Set, attr.as_str(vm.interns)));
|
|
733
|
+
}
|
|
734
|
+
};
|
|
735
|
+
value.map(CallResult::Value)
|
|
736
|
+
}
|
|
737
|
+
|
|
738
|
+
fn py_sub(
|
|
739
|
+
&self,
|
|
740
|
+
_other: &Self,
|
|
741
|
+
_vm: &mut VM<'_, '_, impl ResourceTracker>,
|
|
742
|
+
) -> Result<Option<Value>, crate::resource::ResourceError> {
|
|
743
|
+
// This is called from heap.rs with two Sets
|
|
744
|
+
// We need interns for contains check, but py_sub doesn't have it
|
|
745
|
+
// This is a limitation - we'll need to handle this differently
|
|
746
|
+
// For now, return None to indicate not supported via this path
|
|
747
|
+
Ok(None)
|
|
748
|
+
}
|
|
749
|
+
}
|
|
750
|
+
|
|
751
|
+
/// Pure set/frozenset binary operators shared by both concrete container types.
|
|
752
|
+
#[derive(Debug, Clone, Copy)]
|
|
753
|
+
pub(crate) enum SetBinaryOp {
|
|
754
|
+
And,
|
|
755
|
+
Or,
|
|
756
|
+
Xor,
|
|
757
|
+
Sub,
|
|
758
|
+
}
|
|
759
|
+
|
|
760
|
+
/// Helper methods for set operations with arbitrary iterables.
|
|
761
|
+
impl Set {
|
|
762
|
+
/// Implements operator-form set algebra, which only accepts set/frozenset operands.
|
|
763
|
+
///
|
|
764
|
+
/// Unlike method forms such as `set.union(iterable)`, the binary operators
|
|
765
|
+
/// `& | ^ -` are intentionally strict and return `None` for operands outside
|
|
766
|
+
/// the set-like values CPython accepts here (`set`, `frozenset`,
|
|
767
|
+
/// `dict_keys`, and `dict_items`) so the VM can raise the standard
|
|
768
|
+
/// unsupported-operands `TypeError`.
|
|
769
|
+
pub(crate) fn binary_op_value(
|
|
770
|
+
&self,
|
|
771
|
+
other: &Value,
|
|
772
|
+
op: SetBinaryOp,
|
|
773
|
+
vm: &mut VM<'_, '_, impl ResourceTracker>,
|
|
774
|
+
) -> RunResult<Option<Self>> {
|
|
775
|
+
let Some(other_storage) = get_storage_from_set_operand(other, vm)? else {
|
|
776
|
+
return Ok(None);
|
|
777
|
+
};
|
|
778
|
+
defer_drop!(other_storage, vm);
|
|
779
|
+
|
|
780
|
+
let result = match op {
|
|
781
|
+
SetBinaryOp::And => Self(self.0.intersection(other_storage, vm)?),
|
|
782
|
+
SetBinaryOp::Or => Self(self.0.union(other_storage, vm)?),
|
|
783
|
+
SetBinaryOp::Xor => Self(self.0.symmetric_difference(other_storage, vm)?),
|
|
784
|
+
SetBinaryOp::Sub => Self(self.0.difference(other_storage, vm)?),
|
|
785
|
+
};
|
|
786
|
+
Ok(Some(result))
|
|
787
|
+
}
|
|
788
|
+
|
|
789
|
+
/// Updates this set with elements from an iterable value.
|
|
790
|
+
fn update_from_value(&mut self, other: Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<()> {
|
|
791
|
+
let heap = &mut *vm.heap;
|
|
792
|
+
// Try to get entries from a Set/FrozenSet directly
|
|
793
|
+
let entries_opt = match &other {
|
|
794
|
+
Value::Ref(id) => match heap.get(*id) {
|
|
795
|
+
HeapData::Set(other_set) => Some(other_set.0.clone_entries(heap)),
|
|
796
|
+
HeapData::FrozenSet(other_set) => Some(other_set.0.clone_entries(heap)),
|
|
797
|
+
_ => None,
|
|
798
|
+
},
|
|
799
|
+
_ => None,
|
|
800
|
+
};
|
|
801
|
+
|
|
802
|
+
if let Some(entries) = entries_opt {
|
|
803
|
+
other.drop_with_heap(heap);
|
|
804
|
+
for (value, _hash) in entries {
|
|
805
|
+
self.add(value, vm)?;
|
|
806
|
+
}
|
|
807
|
+
return Ok(());
|
|
808
|
+
}
|
|
809
|
+
|
|
810
|
+
// Fall back to creating a temporary set from the iterable
|
|
811
|
+
let temp_set = Self::from_iterable(other, vm)?;
|
|
812
|
+
defer_drop!(temp_set, vm);
|
|
813
|
+
self.0.update(&temp_set.0, vm)?;
|
|
814
|
+
Ok(())
|
|
815
|
+
}
|
|
816
|
+
|
|
817
|
+
/// Returns a new set with elements from both this set and an iterable.
|
|
818
|
+
fn union_from_value(&self, other: Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<Self> {
|
|
819
|
+
let other_storage = Self::get_storage_from_value(other, vm)?;
|
|
820
|
+
defer_drop!(other_storage, vm);
|
|
821
|
+
let result_storage = self.0.union(other_storage, vm)?;
|
|
822
|
+
Ok(Self(result_storage))
|
|
823
|
+
}
|
|
824
|
+
|
|
825
|
+
/// Returns a new set with elements common to both this set and an iterable.
|
|
826
|
+
fn intersection_from_value(&self, other: Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<Self> {
|
|
827
|
+
let other_storage = Self::get_storage_from_value(other, vm)?;
|
|
828
|
+
defer_drop!(other_storage, vm);
|
|
829
|
+
let result_storage = self.0.intersection(other_storage, vm)?;
|
|
830
|
+
Ok(Self(result_storage))
|
|
831
|
+
}
|
|
832
|
+
|
|
833
|
+
/// Returns a new set with elements in this set but not in an iterable.
|
|
834
|
+
fn difference_from_value(&self, other: Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<Self> {
|
|
835
|
+
let other_storage = Self::get_storage_from_value(other, vm)?;
|
|
836
|
+
defer_drop!(other_storage, vm);
|
|
837
|
+
let result_storage = self.0.difference(other_storage, vm)?;
|
|
838
|
+
Ok(Self(result_storage))
|
|
839
|
+
}
|
|
840
|
+
|
|
841
|
+
/// Returns a new set with elements in either set but not both.
|
|
842
|
+
fn symmetric_difference_from_value(
|
|
843
|
+
&self,
|
|
844
|
+
other: Value,
|
|
845
|
+
vm: &mut VM<'_, '_, impl ResourceTracker>,
|
|
846
|
+
) -> RunResult<Self> {
|
|
847
|
+
let other_storage = Self::get_storage_from_value(other, vm)?;
|
|
848
|
+
defer_drop!(other_storage, vm);
|
|
849
|
+
let result_storage = self.0.symmetric_difference(other_storage, vm)?;
|
|
850
|
+
Ok(Self(result_storage))
|
|
851
|
+
}
|
|
852
|
+
|
|
853
|
+
/// Checks if this set is a subset of an iterable.
|
|
854
|
+
fn issubset_from_value(&self, other: &Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<bool> {
|
|
855
|
+
// Try to get entries from a Set/FrozenSet directly
|
|
856
|
+
let entries_opt = match other {
|
|
857
|
+
Value::Ref(id) => match vm.heap.get(*id) {
|
|
858
|
+
HeapData::Set(other_set) => Some(other_set.0.clone_entries(vm.heap)),
|
|
859
|
+
HeapData::FrozenSet(other_set) => Some(other_set.0.clone_entries(vm.heap)),
|
|
860
|
+
_ => None,
|
|
861
|
+
},
|
|
862
|
+
_ => None,
|
|
863
|
+
};
|
|
864
|
+
|
|
865
|
+
if let Some(entries) = entries_opt {
|
|
866
|
+
let other_storage = SetStorage::from_entries(entries);
|
|
867
|
+
defer_drop!(other_storage, vm);
|
|
868
|
+
return self.0.is_subset(other_storage, vm);
|
|
869
|
+
}
|
|
870
|
+
|
|
871
|
+
// Handle all other iterables (list, tuple, range, str, bytes, dict, etc.)
|
|
872
|
+
let temp = Self::from_iterable(other.clone_with_heap(vm), vm)?;
|
|
873
|
+
defer_drop!(temp, vm);
|
|
874
|
+
self.0.is_subset(&temp.0, vm)
|
|
875
|
+
}
|
|
876
|
+
|
|
877
|
+
/// Checks if this set is a superset of an iterable.
|
|
878
|
+
fn issuperset_from_value(&self, other: &Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<bool> {
|
|
879
|
+
// Try to get entries from a Set/FrozenSet directly
|
|
880
|
+
let entries_opt = match other {
|
|
881
|
+
Value::Ref(id) => match vm.heap.get(*id) {
|
|
882
|
+
HeapData::Set(other_set) => Some(other_set.0.clone_entries(vm.heap)),
|
|
883
|
+
HeapData::FrozenSet(other_set) => Some(other_set.0.clone_entries(vm.heap)),
|
|
884
|
+
_ => None,
|
|
885
|
+
},
|
|
886
|
+
_ => None,
|
|
887
|
+
};
|
|
888
|
+
|
|
889
|
+
if let Some(entries) = entries_opt {
|
|
890
|
+
let other_storage = SetStorage::from_entries(entries);
|
|
891
|
+
defer_drop!(other_storage, vm);
|
|
892
|
+
return self.0.is_superset(other_storage, vm);
|
|
893
|
+
}
|
|
894
|
+
|
|
895
|
+
// Handle all other iterables (list, tuple, range, str, bytes, dict, etc.)
|
|
896
|
+
let temp = Self::from_iterable(other.clone_with_heap(vm), vm)?;
|
|
897
|
+
defer_drop!(temp, vm);
|
|
898
|
+
self.0.is_superset(&temp.0, vm)
|
|
899
|
+
}
|
|
900
|
+
|
|
901
|
+
/// Checks if this set has no elements in common with an iterable.
|
|
902
|
+
fn isdisjoint_from_value(&self, other: &Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<bool> {
|
|
903
|
+
// Try to get entries from a Set/FrozenSet directly
|
|
904
|
+
let entries_opt = match other {
|
|
905
|
+
Value::Ref(id) => match vm.heap.get(*id) {
|
|
906
|
+
HeapData::Set(other_set) => Some(other_set.0.clone_entries(vm.heap)),
|
|
907
|
+
HeapData::FrozenSet(other_set) => Some(other_set.0.clone_entries(vm.heap)),
|
|
908
|
+
_ => None,
|
|
909
|
+
},
|
|
910
|
+
_ => None,
|
|
911
|
+
};
|
|
912
|
+
|
|
913
|
+
if let Some(entries) = entries_opt {
|
|
914
|
+
let other_storage = SetStorage::from_entries(entries);
|
|
915
|
+
defer_drop!(other_storage, vm);
|
|
916
|
+
return self.0.is_disjoint(other_storage, vm);
|
|
917
|
+
}
|
|
918
|
+
|
|
919
|
+
// Handle all other iterables (list, tuple, range, str, bytes, dict, etc.)
|
|
920
|
+
let temp = Self::from_iterable(other.clone_with_heap(vm), vm)?;
|
|
921
|
+
defer_drop!(temp, vm);
|
|
922
|
+
self.0.is_disjoint(&temp.0, vm)
|
|
923
|
+
}
|
|
924
|
+
|
|
925
|
+
/// Helper to get SetStorage from a Value (either directly or by conversion).
|
|
926
|
+
fn get_storage_from_value(value: Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<SetStorage> {
|
|
927
|
+
// Try to get entries from a Set/FrozenSet directly
|
|
928
|
+
let entries_opt = match &value {
|
|
929
|
+
Value::Ref(id) => match vm.heap.get(*id) {
|
|
930
|
+
HeapData::Set(set) => Some(set.0.clone_entries(vm.heap)),
|
|
931
|
+
HeapData::FrozenSet(set) => Some(set.0.clone_entries(vm.heap)),
|
|
932
|
+
_ => None,
|
|
933
|
+
},
|
|
934
|
+
_ => None,
|
|
935
|
+
};
|
|
936
|
+
|
|
937
|
+
if let Some(entries) = entries_opt {
|
|
938
|
+
value.drop_with_heap(vm);
|
|
939
|
+
return Ok(SetStorage::from_entries(entries));
|
|
940
|
+
}
|
|
941
|
+
|
|
942
|
+
// Convert iterable to set
|
|
943
|
+
let temp_set = Self::from_iterable(value, vm)?;
|
|
944
|
+
Ok(temp_set.0)
|
|
945
|
+
}
|
|
946
|
+
}
|
|
947
|
+
|
|
948
|
+
/// Python frozenset type - immutable, unordered collection of unique hashable elements.
|
|
949
|
+
///
|
|
950
|
+
/// FrozenSets support the same set algebra operations as sets (union, intersection,
|
|
951
|
+
/// difference, symmetric difference) but are immutable and therefore hashable.
|
|
952
|
+
///
|
|
953
|
+
/// # Hashability
|
|
954
|
+
/// Unlike mutable sets, frozensets can be used as dict keys or set elements because
|
|
955
|
+
/// they are immutable. The hash is computed as the XOR of element hashes (order-independent).
|
|
956
|
+
#[derive(Debug, Default)]
|
|
957
|
+
pub(crate) struct FrozenSet(SetStorage);
|
|
958
|
+
|
|
959
|
+
impl FrozenSet {
|
|
960
|
+
/// Creates a new empty frozenset.
|
|
961
|
+
#[must_use]
|
|
962
|
+
pub fn new() -> Self {
|
|
963
|
+
Self(SetStorage::new())
|
|
964
|
+
}
|
|
965
|
+
|
|
966
|
+
/// Returns the number of elements in the frozenset.
|
|
967
|
+
#[must_use]
|
|
968
|
+
pub fn len(&self) -> usize {
|
|
969
|
+
self.0.len()
|
|
970
|
+
}
|
|
971
|
+
|
|
972
|
+
/// Returns true if the frozenset is empty.
|
|
973
|
+
#[must_use]
|
|
974
|
+
pub fn is_empty(&self) -> bool {
|
|
975
|
+
self.0.is_empty()
|
|
976
|
+
}
|
|
977
|
+
|
|
978
|
+
/// Returns whether this frozenset contains any heap references (`Value::Ref`).
|
|
979
|
+
///
|
|
980
|
+
/// Used during allocation to determine if this container could create cycles.
|
|
981
|
+
#[inline]
|
|
982
|
+
#[must_use]
|
|
983
|
+
pub fn has_refs(&self) -> bool {
|
|
984
|
+
self.0.has_refs()
|
|
985
|
+
}
|
|
986
|
+
|
|
987
|
+
/// Returns a shallow copy of the frozenset.
|
|
988
|
+
#[must_use]
|
|
989
|
+
pub fn copy(&self, heap: &mut Heap<impl ResourceTracker>) -> Self {
|
|
990
|
+
Self(self.0.clone_with_heap(heap))
|
|
991
|
+
}
|
|
992
|
+
|
|
993
|
+
/// Returns the internal storage.
|
|
994
|
+
pub(crate) fn storage(&self) -> &SetStorage {
|
|
995
|
+
&self.0
|
|
996
|
+
}
|
|
997
|
+
|
|
998
|
+
/// Checks if the frozenset contains a value.
|
|
999
|
+
pub fn contains(&self, value: &Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<bool> {
|
|
1000
|
+
self.0.contains(value, vm)
|
|
1001
|
+
}
|
|
1002
|
+
|
|
1003
|
+
/// Computes the hash of this frozenset.
|
|
1004
|
+
///
|
|
1005
|
+
/// The hash is the XOR of all element hashes, making it order-independent.
|
|
1006
|
+
/// Checks recursion depth before recursing into element hashes.
|
|
1007
|
+
pub fn compute_hash(
|
|
1008
|
+
&self,
|
|
1009
|
+
heap: &mut Heap<impl ResourceTracker>,
|
|
1010
|
+
interns: &Interns,
|
|
1011
|
+
) -> Result<Option<u64>, ResourceError> {
|
|
1012
|
+
let token = heap.incr_recursion_depth()?;
|
|
1013
|
+
defer_drop!(token, heap);
|
|
1014
|
+
let mut hash: u64 = 0;
|
|
1015
|
+
for entry in &self.0.entries {
|
|
1016
|
+
// All elements must be hashable (enforced at construction)
|
|
1017
|
+
match entry.value.py_hash(heap, interns)? {
|
|
1018
|
+
Some(h) => hash ^= h,
|
|
1019
|
+
None => return Ok(None),
|
|
1020
|
+
}
|
|
1021
|
+
}
|
|
1022
|
+
Ok(Some(hash))
|
|
1023
|
+
}
|
|
1024
|
+
|
|
1025
|
+
/// Creates a frozenset from a Set, consuming the Set's storage.
|
|
1026
|
+
///
|
|
1027
|
+
/// This is used when we need to convert a mutable set to an immutable frozenset
|
|
1028
|
+
/// without cloning.
|
|
1029
|
+
pub fn from_set(set: Set) -> Self {
|
|
1030
|
+
Self(set.0)
|
|
1031
|
+
}
|
|
1032
|
+
|
|
1033
|
+
/// Creates a frozenset from the `frozenset()` constructor call.
|
|
1034
|
+
///
|
|
1035
|
+
/// - `frozenset()` with no args returns an empty frozenset
|
|
1036
|
+
/// - `frozenset(iterable)` creates a frozenset from any iterable (list, tuple, set, dict, range, str, bytes)
|
|
1037
|
+
pub fn init(vm: &mut VM<'_, '_, impl ResourceTracker>, args: ArgValues) -> RunResult<Value> {
|
|
1038
|
+
let value = args.get_zero_one_arg("frozenset", vm.heap)?;
|
|
1039
|
+
let frozenset = match value {
|
|
1040
|
+
None => Self::new(),
|
|
1041
|
+
Some(v) => Self::from_set(Set::from_iterable(v, vm)?),
|
|
1042
|
+
};
|
|
1043
|
+
let heap_id = vm.heap.allocate(HeapData::FrozenSet(frozenset))?;
|
|
1044
|
+
Ok(Value::Ref(heap_id))
|
|
1045
|
+
}
|
|
1046
|
+
|
|
1047
|
+
/// Returns a new frozenset with elements from both this and another set.
|
|
1048
|
+
pub(crate) fn union(&self, other: &SetStorage, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<Self> {
|
|
1049
|
+
Ok(Self(self.0.union(other, vm)?))
|
|
1050
|
+
}
|
|
1051
|
+
|
|
1052
|
+
/// Returns a new frozenset with elements common to both sets.
|
|
1053
|
+
pub(crate) fn intersection(
|
|
1054
|
+
&self,
|
|
1055
|
+
other: &SetStorage,
|
|
1056
|
+
vm: &mut VM<'_, '_, impl ResourceTracker>,
|
|
1057
|
+
) -> RunResult<Self> {
|
|
1058
|
+
Ok(Self(self.0.intersection(other, vm)?))
|
|
1059
|
+
}
|
|
1060
|
+
|
|
1061
|
+
/// Returns a new frozenset with elements in this set but not in other.
|
|
1062
|
+
pub(crate) fn difference(&self, other: &SetStorage, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<Self> {
|
|
1063
|
+
Ok(Self(self.0.difference(other, vm)?))
|
|
1064
|
+
}
|
|
1065
|
+
|
|
1066
|
+
/// Returns a new frozenset with elements in either set but not both.
|
|
1067
|
+
pub(crate) fn symmetric_difference(
|
|
1068
|
+
&self,
|
|
1069
|
+
other: &SetStorage,
|
|
1070
|
+
vm: &mut VM<'_, '_, impl ResourceTracker>,
|
|
1071
|
+
) -> RunResult<Self> {
|
|
1072
|
+
Ok(Self(self.0.symmetric_difference(other, vm)?))
|
|
1073
|
+
}
|
|
1074
|
+
}
|
|
1075
|
+
|
|
1076
|
+
impl PyTrait for FrozenSet {
|
|
1077
|
+
fn py_type(&self, _heap: &Heap<impl ResourceTracker>) -> Type {
|
|
1078
|
+
Type::FrozenSet
|
|
1079
|
+
}
|
|
1080
|
+
|
|
1081
|
+
fn py_estimate_size(&self) -> usize {
|
|
1082
|
+
self.0.estimate_size()
|
|
1083
|
+
}
|
|
1084
|
+
|
|
1085
|
+
fn py_len(&self, _vm: &VM<'_, '_, impl ResourceTracker>) -> Option<usize> {
|
|
1086
|
+
Some(self.len())
|
|
1087
|
+
}
|
|
1088
|
+
|
|
1089
|
+
fn py_eq(&self, other: &Self, vm: &mut VM<'_, '_, impl ResourceTracker>) -> Result<bool, ResourceError> {
|
|
1090
|
+
if self.len() != other.len() {
|
|
1091
|
+
return Ok(false);
|
|
1092
|
+
}
|
|
1093
|
+
let token = vm.heap.incr_recursion_depth()?;
|
|
1094
|
+
defer_drop!(token, vm);
|
|
1095
|
+
self.0.eq(&other.0, vm)
|
|
1096
|
+
}
|
|
1097
|
+
|
|
1098
|
+
fn py_dec_ref_ids(&mut self, stack: &mut Vec<HeapId>) {
|
|
1099
|
+
self.0.collect_dec_ref_ids(stack);
|
|
1100
|
+
}
|
|
1101
|
+
|
|
1102
|
+
fn py_bool(&self, _vm: &VM<'_, '_, impl ResourceTracker>) -> bool {
|
|
1103
|
+
!self.is_empty()
|
|
1104
|
+
}
|
|
1105
|
+
|
|
1106
|
+
fn py_repr_fmt(
|
|
1107
|
+
&self,
|
|
1108
|
+
f: &mut impl Write,
|
|
1109
|
+
vm: &VM<'_, '_, impl ResourceTracker>,
|
|
1110
|
+
heap_ids: &mut AHashSet<HeapId>,
|
|
1111
|
+
) -> std::fmt::Result {
|
|
1112
|
+
self.0.repr_fmt(f, vm, heap_ids, "frozenset")
|
|
1113
|
+
}
|
|
1114
|
+
|
|
1115
|
+
fn py_call_attr(
|
|
1116
|
+
&mut self,
|
|
1117
|
+
_self_id: HeapId,
|
|
1118
|
+
vm: &mut VM<'_, '_, impl ResourceTracker>,
|
|
1119
|
+
attr: &EitherStr,
|
|
1120
|
+
args: ArgValues,
|
|
1121
|
+
) -> RunResult<CallResult> {
|
|
1122
|
+
let heap = &mut *vm.heap;
|
|
1123
|
+
let interns = vm.interns;
|
|
1124
|
+
let value = match attr.static_string() {
|
|
1125
|
+
Some(StaticStrings::Copy) => {
|
|
1126
|
+
args.check_zero_args("frozenset.copy", heap)?;
|
|
1127
|
+
let copy = self.copy(heap);
|
|
1128
|
+
let heap_id = heap.allocate(HeapData::FrozenSet(copy))?;
|
|
1129
|
+
Ok(Value::Ref(heap_id))
|
|
1130
|
+
}
|
|
1131
|
+
Some(StaticStrings::Union) => {
|
|
1132
|
+
let other = args.get_one_arg("frozenset.union", heap)?;
|
|
1133
|
+
let other_storage = Set::get_storage_from_value(other, vm)?;
|
|
1134
|
+
defer_drop!(other_storage, vm);
|
|
1135
|
+
let result = self.union(other_storage, vm)?;
|
|
1136
|
+
let heap_id = vm.heap.allocate(HeapData::FrozenSet(result))?;
|
|
1137
|
+
Ok(Value::Ref(heap_id))
|
|
1138
|
+
}
|
|
1139
|
+
Some(StaticStrings::Intersection) => {
|
|
1140
|
+
let other = args.get_one_arg("frozenset.intersection", heap)?;
|
|
1141
|
+
let other_storage = Set::get_storage_from_value(other, vm)?;
|
|
1142
|
+
defer_drop!(other_storage, vm);
|
|
1143
|
+
let result = self.intersection(other_storage, vm)?;
|
|
1144
|
+
let heap_id = vm.heap.allocate(HeapData::FrozenSet(result))?;
|
|
1145
|
+
Ok(Value::Ref(heap_id))
|
|
1146
|
+
}
|
|
1147
|
+
Some(StaticStrings::Difference) => {
|
|
1148
|
+
let other = args.get_one_arg("frozenset.difference", heap)?;
|
|
1149
|
+
let other_storage = Set::get_storage_from_value(other, vm)?;
|
|
1150
|
+
defer_drop!(other_storage, vm);
|
|
1151
|
+
let result = self.difference(other_storage, vm)?;
|
|
1152
|
+
let heap_id = vm.heap.allocate(HeapData::FrozenSet(result))?;
|
|
1153
|
+
Ok(Value::Ref(heap_id))
|
|
1154
|
+
}
|
|
1155
|
+
Some(StaticStrings::SymmetricDifference) => {
|
|
1156
|
+
let other = args.get_one_arg("frozenset.symmetric_difference", heap)?;
|
|
1157
|
+
let other_storage = Set::get_storage_from_value(other, vm)?;
|
|
1158
|
+
defer_drop!(other_storage, vm);
|
|
1159
|
+
let result = self.symmetric_difference(other_storage, vm)?;
|
|
1160
|
+
let heap_id = vm.heap.allocate(HeapData::FrozenSet(result))?;
|
|
1161
|
+
Ok(Value::Ref(heap_id))
|
|
1162
|
+
}
|
|
1163
|
+
Some(StaticStrings::Issubset) => {
|
|
1164
|
+
let other = args.get_one_arg("frozenset.issubset", heap)?;
|
|
1165
|
+
defer_drop!(other, vm);
|
|
1166
|
+
Ok(Value::Bool(self.issubset_from_value(other, vm)?))
|
|
1167
|
+
}
|
|
1168
|
+
Some(StaticStrings::Issuperset) => {
|
|
1169
|
+
let other = args.get_one_arg("frozenset.issuperset", heap)?;
|
|
1170
|
+
defer_drop!(other, vm);
|
|
1171
|
+
Ok(Value::Bool(self.issuperset_from_value(other, vm)?))
|
|
1172
|
+
}
|
|
1173
|
+
Some(StaticStrings::Isdisjoint) => {
|
|
1174
|
+
let other = args.get_one_arg("frozenset.isdisjoint", heap)?;
|
|
1175
|
+
defer_drop!(other, vm);
|
|
1176
|
+
Ok(Value::Bool(self.isdisjoint_from_value(other, vm)?))
|
|
1177
|
+
}
|
|
1178
|
+
_ => {
|
|
1179
|
+
args.drop_with_heap(heap);
|
|
1180
|
+
return Err(ExcType::attribute_error(Type::FrozenSet, attr.as_str(interns)));
|
|
1181
|
+
}
|
|
1182
|
+
};
|
|
1183
|
+
value.map(CallResult::Value)
|
|
1184
|
+
}
|
|
1185
|
+
|
|
1186
|
+
fn py_sub(
|
|
1187
|
+
&self,
|
|
1188
|
+
_other: &Self,
|
|
1189
|
+
_vm: &mut VM<'_, '_, impl ResourceTracker>,
|
|
1190
|
+
) -> Result<Option<Value>, crate::resource::ResourceError> {
|
|
1191
|
+
// Same limitation as Set - needs interns
|
|
1192
|
+
Ok(None)
|
|
1193
|
+
}
|
|
1194
|
+
}
|
|
1195
|
+
|
|
1196
|
+
/// Helper methods for frozenset operations with arbitrary iterables.
|
|
1197
|
+
impl FrozenSet {
|
|
1198
|
+
/// Implements operator-form set algebra, which only accepts set/frozenset operands.
|
|
1199
|
+
///
|
|
1200
|
+
/// CPython returns the type of the left operand for pure set/frozenset binary
|
|
1201
|
+
/// operators, so this helper keeps the result as `frozenset` even when the
|
|
1202
|
+
/// right operand is a mutable `set`. Like `set`, the accepted right-hand
|
|
1203
|
+
/// side includes CPython's set-like dict views.
|
|
1204
|
+
pub(crate) fn binary_op_value(
|
|
1205
|
+
&self,
|
|
1206
|
+
other: &Value,
|
|
1207
|
+
op: SetBinaryOp,
|
|
1208
|
+
vm: &mut VM<'_, '_, impl ResourceTracker>,
|
|
1209
|
+
) -> RunResult<Option<Self>> {
|
|
1210
|
+
let Some(other_storage) = get_storage_from_set_operand(other, vm)? else {
|
|
1211
|
+
return Ok(None);
|
|
1212
|
+
};
|
|
1213
|
+
defer_drop!(other_storage, vm);
|
|
1214
|
+
|
|
1215
|
+
let result = match op {
|
|
1216
|
+
SetBinaryOp::And => Self(self.0.intersection(other_storage, vm)?),
|
|
1217
|
+
SetBinaryOp::Or => Self(self.0.union(other_storage, vm)?),
|
|
1218
|
+
SetBinaryOp::Xor => Self(self.0.symmetric_difference(other_storage, vm)?),
|
|
1219
|
+
SetBinaryOp::Sub => Self(self.0.difference(other_storage, vm)?),
|
|
1220
|
+
};
|
|
1221
|
+
Ok(Some(result))
|
|
1222
|
+
}
|
|
1223
|
+
|
|
1224
|
+
/// Checks if this frozenset is a subset of an iterable.
|
|
1225
|
+
fn issubset_from_value(&self, other: &Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<bool> {
|
|
1226
|
+
// Try to get entries from a Set/FrozenSet directly
|
|
1227
|
+
let entries_opt = match other {
|
|
1228
|
+
Value::Ref(id) => match vm.heap.get(*id) {
|
|
1229
|
+
HeapData::Set(other_set) => Some(other_set.0.clone_entries(vm.heap)),
|
|
1230
|
+
HeapData::FrozenSet(other_set) => Some(other_set.0.clone_entries(vm.heap)),
|
|
1231
|
+
_ => None,
|
|
1232
|
+
},
|
|
1233
|
+
_ => None,
|
|
1234
|
+
};
|
|
1235
|
+
|
|
1236
|
+
if let Some(entries) = entries_opt {
|
|
1237
|
+
// Build temporary storage and check
|
|
1238
|
+
let other_storage = SetStorage::from_entries(entries);
|
|
1239
|
+
defer_drop!(other_storage, vm);
|
|
1240
|
+
return self.0.is_subset(other_storage, vm);
|
|
1241
|
+
}
|
|
1242
|
+
|
|
1243
|
+
// Handle all other iterables (list, tuple, range, str, bytes, dict, etc.)
|
|
1244
|
+
let temp = Set::from_iterable(other.clone_with_heap(vm), vm)?;
|
|
1245
|
+
defer_drop!(temp, vm);
|
|
1246
|
+
self.0.is_subset(&temp.0, vm)
|
|
1247
|
+
}
|
|
1248
|
+
|
|
1249
|
+
/// Checks if this frozenset is a superset of an iterable.
|
|
1250
|
+
fn issuperset_from_value(&self, other: &Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<bool> {
|
|
1251
|
+
// Try to get entries from a Set/FrozenSet directly
|
|
1252
|
+
let entries_opt = match other {
|
|
1253
|
+
Value::Ref(id) => match vm.heap.get(*id) {
|
|
1254
|
+
HeapData::Set(other_set) => Some(other_set.0.clone_entries(vm.heap)),
|
|
1255
|
+
HeapData::FrozenSet(other_set) => Some(other_set.0.clone_entries(vm.heap)),
|
|
1256
|
+
_ => None,
|
|
1257
|
+
},
|
|
1258
|
+
_ => None,
|
|
1259
|
+
};
|
|
1260
|
+
|
|
1261
|
+
if let Some(entries) = entries_opt {
|
|
1262
|
+
// Build temporary storage and check
|
|
1263
|
+
let other_storage = SetStorage::from_entries(entries);
|
|
1264
|
+
defer_drop!(other_storage, vm);
|
|
1265
|
+
return self.0.is_superset(other_storage, vm);
|
|
1266
|
+
}
|
|
1267
|
+
|
|
1268
|
+
// Handle all other iterables (list, tuple, range, str, bytes, dict, etc.)
|
|
1269
|
+
let temp = Set::from_iterable(other.clone_with_heap(vm), vm)?;
|
|
1270
|
+
defer_drop!(temp, vm);
|
|
1271
|
+
self.0.is_superset(&temp.0, vm)
|
|
1272
|
+
}
|
|
1273
|
+
|
|
1274
|
+
/// Checks if this frozenset has no elements in common with an iterable.
|
|
1275
|
+
fn isdisjoint_from_value(&self, other: &Value, vm: &mut VM<'_, '_, impl ResourceTracker>) -> RunResult<bool> {
|
|
1276
|
+
// Try to get entries from a Set/FrozenSet directly
|
|
1277
|
+
let entries_opt = match other {
|
|
1278
|
+
Value::Ref(id) => match vm.heap.get(*id) {
|
|
1279
|
+
HeapData::Set(other_set) => Some(other_set.0.clone_entries(vm.heap)),
|
|
1280
|
+
HeapData::FrozenSet(other_set) => Some(other_set.0.clone_entries(vm.heap)),
|
|
1281
|
+
_ => None,
|
|
1282
|
+
},
|
|
1283
|
+
_ => None,
|
|
1284
|
+
};
|
|
1285
|
+
|
|
1286
|
+
if let Some(entries) = entries_opt {
|
|
1287
|
+
// Build temporary storage and check
|
|
1288
|
+
let other_storage = SetStorage::from_entries(entries);
|
|
1289
|
+
defer_drop!(other_storage, vm);
|
|
1290
|
+
return self.0.is_disjoint(other_storage, vm);
|
|
1291
|
+
}
|
|
1292
|
+
|
|
1293
|
+
// Handle all other iterables (list, tuple, range, str, bytes, dict, etc.)
|
|
1294
|
+
let temp = Set::from_iterable(other.clone_with_heap(vm), vm)?;
|
|
1295
|
+
defer_drop!(temp, vm);
|
|
1296
|
+
self.0.is_disjoint(&temp.0, vm)
|
|
1297
|
+
}
|
|
1298
|
+
}
|
|
1299
|
+
|
|
1300
|
+
/// Returns temporary set storage only for operator-valid set operands.
|
|
1301
|
+
///
|
|
1302
|
+
/// This is stricter than `Set::get_storage_from_value(...)`: operator forms
|
|
1303
|
+
/// only accept CPython's set-like operands (`set`, `frozenset`, `dict_keys`,
|
|
1304
|
+
/// and `dict_items`), while method forms accept any iterable.
|
|
1305
|
+
fn get_storage_from_set_operand(
|
|
1306
|
+
value: &Value,
|
|
1307
|
+
vm: &mut VM<'_, '_, impl ResourceTracker>,
|
|
1308
|
+
) -> RunResult<Option<SetStorage>> {
|
|
1309
|
+
let Value::Ref(id) = value else {
|
|
1310
|
+
return Ok(None);
|
|
1311
|
+
};
|
|
1312
|
+
|
|
1313
|
+
match vm.heap.get(*id) {
|
|
1314
|
+
HeapData::Set(set) => Ok(Some(SetStorage::from_entries(set.0.clone_entries(vm.heap)))),
|
|
1315
|
+
HeapData::FrozenSet(set) => Ok(Some(SetStorage::from_entries(set.0.clone_entries(vm.heap)))),
|
|
1316
|
+
// Dict views are `Copy` — matched value is not borrowed from the heap,
|
|
1317
|
+
// so `to_set` can take `&mut VM` below without conflict.
|
|
1318
|
+
HeapData::DictKeysView(view) => {
|
|
1319
|
+
let Set(storage) = view.to_set(vm)?;
|
|
1320
|
+
Ok(Some(storage))
|
|
1321
|
+
}
|
|
1322
|
+
HeapData::DictItemsView(view) => {
|
|
1323
|
+
let Set(storage) = view.to_set(vm)?;
|
|
1324
|
+
Ok(Some(storage))
|
|
1325
|
+
}
|
|
1326
|
+
_ => Ok(None),
|
|
1327
|
+
}
|
|
1328
|
+
}
|
|
1329
|
+
|
|
1330
|
+
// Custom serde implementations for SetStorage, Set, and FrozenSet.
|
|
1331
|
+
// Only serialize entries; rebuild the indices hash table on deserialize.
|
|
1332
|
+
|
|
1333
|
+
impl serde::Serialize for SetStorage {
|
|
1334
|
+
fn serialize<S: serde::Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
|
|
1335
|
+
self.entries.serialize(serializer)
|
|
1336
|
+
}
|
|
1337
|
+
}
|
|
1338
|
+
|
|
1339
|
+
impl<'de> serde::Deserialize<'de> for SetStorage {
|
|
1340
|
+
fn deserialize<D: serde::Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
|
|
1341
|
+
let entries: Vec<SetEntry> = serde::Deserialize::deserialize(deserializer)?;
|
|
1342
|
+
// Rebuild the indices hash table from the entries
|
|
1343
|
+
let mut indices = HashTable::with_capacity(entries.len());
|
|
1344
|
+
for (idx, entry) in entries.iter().enumerate() {
|
|
1345
|
+
indices.insert_unique(entry.hash, idx, |&i| entries[i].hash);
|
|
1346
|
+
}
|
|
1347
|
+
Ok(Self { indices, entries })
|
|
1348
|
+
}
|
|
1349
|
+
}
|
|
1350
|
+
|
|
1351
|
+
impl serde::Serialize for Set {
|
|
1352
|
+
fn serialize<S: serde::Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
|
|
1353
|
+
self.0.serialize(serializer)
|
|
1354
|
+
}
|
|
1355
|
+
}
|
|
1356
|
+
|
|
1357
|
+
impl<'de> serde::Deserialize<'de> for Set {
|
|
1358
|
+
fn deserialize<D: serde::Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
|
|
1359
|
+
Ok(Self(SetStorage::deserialize(deserializer)?))
|
|
1360
|
+
}
|
|
1361
|
+
}
|
|
1362
|
+
|
|
1363
|
+
impl serde::Serialize for FrozenSet {
|
|
1364
|
+
fn serialize<S: serde::Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
|
|
1365
|
+
self.0.serialize(serializer)
|
|
1366
|
+
}
|
|
1367
|
+
}
|
|
1368
|
+
|
|
1369
|
+
impl<'de> serde::Deserialize<'de> for FrozenSet {
|
|
1370
|
+
fn deserialize<D: serde::Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
|
|
1371
|
+
Ok(Self(SetStorage::deserialize(deserializer)?))
|
|
1372
|
+
}
|
|
1373
|
+
}
|