classiq 0.37.1__py3-none-any.whl → 0.65.3__py3-none-any.whl
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.
- classiq/__init__.py +49 -34
- classiq/_analyzer_extras/_ipywidgets_async_extension.py +3 -2
- classiq/_analyzer_extras/interactive_hardware.py +3 -3
- classiq/_internals/api_wrapper.py +241 -95
- classiq/_internals/async_utils.py +2 -77
- classiq/_internals/authentication/auth0.py +26 -10
- classiq/_internals/authentication/authentication.py +11 -0
- classiq/_internals/authentication/device.py +18 -8
- classiq/_internals/authentication/password_manager.py +40 -13
- classiq/_internals/authentication/token_manager.py +11 -6
- classiq/_internals/client.py +106 -37
- classiq/_internals/config.py +3 -4
- classiq/_internals/host_checker.py +38 -15
- classiq/_internals/jobs.py +56 -50
- classiq/_internals/type_validation.py +9 -9
- classiq/analyzer/__init__.py +1 -3
- classiq/analyzer/analyzer.py +50 -47
- classiq/analyzer/analyzer_utilities.py +15 -15
- classiq/analyzer/rb.py +19 -20
- classiq/analyzer/show_interactive_hack.py +30 -7
- classiq/analyzer/url_utils.py +2 -3
- classiq/applications/__init__.py +3 -12
- classiq/applications/chemistry/__init__.py +14 -10
- classiq/applications/chemistry/ansatz_parameters.py +4 -4
- classiq/{applications_model_constructors → applications/chemistry}/chemistry_model_constructor.py +170 -170
- classiq/applications/chemistry/ground_state_problem.py +1 -1
- classiq/applications/combinatorial_helpers/allowed_constraints.py +23 -0
- classiq/applications/combinatorial_helpers/arithmetic/arithmetic_expression.py +35 -0
- classiq/applications/combinatorial_helpers/arithmetic/isolation.py +42 -0
- classiq/applications/combinatorial_helpers/combinatorial_problem_utils.py +166 -0
- classiq/applications/combinatorial_helpers/encoding_mapping.py +107 -0
- classiq/applications/combinatorial_helpers/encoding_utils.py +124 -0
- classiq/applications/combinatorial_helpers/memory.py +75 -0
- classiq/applications/combinatorial_helpers/optimization_model.py +193 -0
- classiq/applications/combinatorial_helpers/pauli_helpers/pauli_sparsing.py +31 -0
- classiq/applications/combinatorial_helpers/pauli_helpers/pauli_utils.py +46 -0
- classiq/applications/combinatorial_helpers/pyomo_utils.py +447 -0
- classiq/applications/combinatorial_helpers/sympy_utils.py +22 -0
- classiq/applications/combinatorial_helpers/transformations/encoding.py +189 -0
- classiq/applications/combinatorial_helpers/transformations/fixed_variables.py +143 -0
- classiq/applications/combinatorial_helpers/transformations/ising_converter.py +120 -0
- classiq/applications/combinatorial_helpers/transformations/penalty.py +31 -0
- classiq/applications/combinatorial_helpers/transformations/penalty_support.py +37 -0
- classiq/applications/combinatorial_helpers/transformations/sign_seperation.py +74 -0
- classiq/applications/combinatorial_helpers/transformations/slack_variables.py +87 -0
- classiq/applications/combinatorial_optimization/__init__.py +24 -5
- classiq/applications/combinatorial_optimization/combinatorial_optimization_config.py +2 -2
- classiq/applications/combinatorial_optimization/combinatorial_optimization_model_constructor.py +137 -0
- classiq/applications/combinatorial_optimization/combinatorial_problem.py +229 -0
- classiq/applications/combinatorial_optimization/examples/__init__.py +1 -3
- classiq/applications/finance/__init__.py +4 -5
- classiq/{applications_model_constructors → applications/finance}/finance_model_constructor.py +50 -57
- classiq/applications/grover/__init__.py +9 -0
- classiq/applications/grover/grover_model_constructor.py +157 -0
- classiq/applications/hamiltonian/__init__.py +0 -0
- classiq/applications/hamiltonian/pauli_decomposition.py +113 -0
- classiq/applications/libraries/__init__.py +0 -0
- classiq/applications/libraries/qmci_library.py +22 -0
- classiq/applications/qnn/__init__.py +2 -4
- classiq/applications/qnn/circuit_utils.py +8 -8
- classiq/applications/qnn/datasets/__init__.py +9 -11
- classiq/applications/qnn/datasets/dataset_base_classes.py +7 -5
- classiq/applications/qnn/datasets/dataset_not.py +2 -1
- classiq/applications/qnn/datasets/dataset_parity.py +2 -2
- classiq/applications/qnn/gradients/quantum_gradient.py +2 -2
- classiq/applications/qnn/gradients/simple_quantum_gradient.py +2 -1
- classiq/applications/qnn/qlayer.py +30 -10
- classiq/applications/qnn/torch_utils.py +4 -3
- classiq/applications/qnn/types.py +7 -7
- classiq/applications/qsvm/__init__.py +6 -4
- classiq/applications/qsvm/qsvm.py +4 -10
- classiq/applications/qsvm/qsvm_data_generation.py +5 -8
- classiq/{applications_model_constructors → applications/qsvm}/qsvm_model_constructor.py +30 -28
- classiq/execution/__init__.py +8 -3
- classiq/execution/all_hardware_devices.py +11 -0
- classiq/execution/execution_session.py +400 -0
- classiq/execution/iqcc.py +63 -0
- classiq/execution/jobs.py +197 -25
- classiq/execution/qnn.py +79 -0
- classiq/executor.py +23 -117
- classiq/interface/_version.py +1 -1
- classiq/interface/analyzer/analysis_params.py +49 -16
- classiq/interface/analyzer/cytoscape_graph.py +15 -9
- classiq/interface/analyzer/result.py +36 -32
- classiq/interface/applications/qsvm.py +28 -25
- classiq/interface/ast_node.py +16 -0
- classiq/interface/backend/backend_preferences.py +390 -119
- classiq/interface/backend/ionq/ionq_quantum_program.py +15 -23
- classiq/interface/backend/pydantic_backend.py +27 -22
- classiq/interface/backend/quantum_backend_providers.py +70 -16
- classiq/interface/chemistry/fermionic_operator.py +43 -32
- classiq/interface/chemistry/ground_state_problem.py +42 -24
- classiq/interface/chemistry/molecule.py +20 -14
- classiq/interface/chemistry/operator.py +75 -236
- classiq/interface/combinatorial_optimization/encoding_types.py +1 -1
- classiq/interface/combinatorial_optimization/examples/ascending_sequence.py +2 -4
- classiq/interface/combinatorial_optimization/examples/greater_than_ilp.py +1 -1
- classiq/interface/combinatorial_optimization/examples/ilp.py +2 -1
- classiq/interface/combinatorial_optimization/examples/integer_portfolio_optimization.py +2 -4
- classiq/interface/combinatorial_optimization/examples/knapsack.py +3 -3
- classiq/interface/combinatorial_optimization/examples/mds.py +2 -1
- classiq/interface/combinatorial_optimization/examples/mht.py +10 -6
- classiq/interface/combinatorial_optimization/examples/mis.py +4 -1
- classiq/interface/combinatorial_optimization/examples/mvc.py +2 -1
- classiq/interface/combinatorial_optimization/examples/portfolio_variations.py +2 -2
- classiq/interface/combinatorial_optimization/examples/set_cover.py +3 -3
- classiq/interface/combinatorial_optimization/examples/tsp.py +4 -3
- classiq/interface/combinatorial_optimization/examples/tsp_digraph.py +6 -2
- classiq/interface/combinatorial_optimization/mht_qaoa_input.py +14 -9
- classiq/interface/combinatorial_optimization/optimization_problem.py +2 -2
- classiq/interface/combinatorial_optimization/result.py +1 -3
- classiq/interface/combinatorial_optimization/solver_types.py +1 -1
- classiq/interface/debug_info/__init__.py +0 -0
- classiq/interface/debug_info/debug_info.py +86 -0
- classiq/interface/exceptions.py +201 -0
- classiq/interface/execution/iqcc.py +19 -0
- classiq/interface/execution/jobs.py +15 -12
- classiq/interface/execution/primitives.py +18 -0
- classiq/interface/executor/constants.py +1 -0
- classiq/interface/executor/estimation.py +2 -2
- classiq/interface/executor/execution_preferences.py +26 -143
- classiq/interface/executor/execution_request.py +36 -56
- classiq/interface/executor/execution_result.py +30 -8
- classiq/interface/executor/iqae_result.py +4 -6
- classiq/interface/executor/optimizer_preferences.py +34 -22
- classiq/interface/executor/{quantum_program.py → quantum_code.py} +44 -34
- classiq/interface/executor/quantum_instruction_set.py +3 -2
- classiq/interface/executor/register_initialization.py +12 -17
- classiq/interface/executor/result.py +122 -64
- classiq/interface/executor/vqe_result.py +11 -11
- classiq/interface/finance/function_input.py +42 -19
- classiq/interface/finance/gaussian_model_input.py +7 -5
- classiq/interface/finance/log_normal_model_input.py +6 -4
- classiq/interface/finance/model_input.py +6 -4
- classiq/interface/generator/adjacency.py +1 -3
- classiq/interface/generator/amplitude_loading.py +27 -14
- classiq/interface/generator/ansatz_library.py +5 -5
- classiq/interface/generator/application_apis/__init__.py +1 -0
- classiq/interface/generator/application_apis/arithmetic_declarations.py +17 -0
- classiq/interface/generator/application_apis/chemistry_declarations.py +27 -187
- classiq/interface/generator/application_apis/combinatorial_optimization_declarations.py +18 -21
- classiq/interface/generator/application_apis/entangler_declarations.py +11 -6
- classiq/interface/generator/application_apis/finance_declarations.py +48 -69
- classiq/interface/generator/application_apis/qsvm_declarations.py +0 -70
- classiq/interface/generator/arith/argument_utils.py +57 -6
- classiq/interface/generator/arith/arithmetic.py +37 -16
- classiq/interface/generator/arith/arithmetic_arg_type_validator.py +15 -17
- classiq/interface/generator/arith/arithmetic_expression_abc.py +70 -26
- classiq/interface/generator/arith/arithmetic_expression_parser.py +18 -12
- classiq/interface/generator/arith/arithmetic_expression_validator.py +61 -43
- classiq/interface/generator/arith/arithmetic_operations.py +19 -16
- classiq/interface/generator/arith/arithmetic_param_getters.py +7 -8
- classiq/interface/generator/arith/arithmetic_result_builder.py +21 -17
- classiq/interface/generator/arith/ast_node_rewrite.py +4 -3
- classiq/interface/generator/arith/binary_ops.py +375 -139
- classiq/interface/generator/arith/endianness.py +1 -1
- classiq/interface/generator/arith/extremum_operations.py +96 -23
- classiq/interface/generator/arith/logical_ops.py +16 -12
- classiq/interface/generator/arith/machine_precision.py +3 -0
- classiq/interface/generator/arith/number_utils.py +44 -48
- classiq/interface/generator/arith/register_user_input.py +70 -27
- classiq/interface/generator/arith/unary_ops.py +57 -46
- classiq/interface/generator/arith/uncomputation_methods.py +1 -1
- classiq/interface/generator/builtin_api_builder.py +2 -9
- classiq/interface/generator/chemistry_function_params.py +5 -5
- classiq/interface/generator/circuit_code/circuit_code.py +7 -7
- classiq/interface/generator/circuit_code/types_and_constants.py +4 -7
- classiq/interface/generator/commuting_pauli_exponentiation.py +8 -6
- classiq/interface/generator/compiler_keywords.py +8 -0
- classiq/interface/generator/complex_type.py +13 -25
- classiq/interface/generator/constant.py +3 -4
- classiq/interface/generator/control_state.py +35 -28
- classiq/interface/generator/copy.py +47 -0
- classiq/interface/generator/custom_ansatz.py +2 -5
- classiq/interface/generator/distance.py +3 -5
- classiq/interface/generator/excitations.py +3 -2
- classiq/interface/generator/expressions/atomic_expression_functions.py +26 -8
- classiq/interface/generator/expressions/enums/__init__.py +0 -10
- classiq/interface/generator/expressions/enums/finance_functions.py +12 -22
- classiq/interface/generator/expressions/evaluated_expression.py +21 -7
- classiq/interface/generator/expressions/expression.py +27 -15
- classiq/interface/generator/expressions/expression_constants.py +9 -3
- classiq/interface/generator/expressions/non_symbolic_expr.py +119 -0
- classiq/interface/generator/expressions/qmod_qarray_proxy.py +99 -0
- classiq/interface/generator/expressions/qmod_qscalar_proxy.py +59 -0
- classiq/interface/generator/expressions/qmod_qstruct_proxy.py +36 -0
- classiq/interface/generator/expressions/qmod_sized_proxy.py +30 -2
- classiq/interface/generator/expressions/qmod_struct_instance.py +14 -2
- classiq/interface/generator/expressions/sympy_supported_expressions.py +20 -11
- classiq/interface/generator/finance.py +3 -3
- classiq/interface/generator/function_param_library.py +6 -6
- classiq/interface/generator/function_param_list_without_self_reference.py +2 -10
- classiq/interface/generator/function_params.py +42 -69
- classiq/interface/generator/functions/__init__.py +0 -22
- classiq/interface/generator/functions/builtins/__init__.py +0 -0
- classiq/interface/generator/functions/builtins/internal_operators.py +16 -0
- classiq/interface/generator/functions/classical_function_declaration.py +18 -9
- classiq/interface/generator/functions/classical_type.py +47 -166
- classiq/interface/generator/functions/concrete_types.py +55 -0
- classiq/interface/generator/functions/function_declaration.py +13 -14
- classiq/interface/generator/functions/port_declaration.py +1 -13
- classiq/interface/generator/functions/qmod_python_interface.py +2 -1
- classiq/interface/generator/functions/type_name.py +90 -0
- classiq/interface/generator/generated_circuit_data.py +155 -22
- classiq/interface/generator/grover_diffuser.py +32 -25
- classiq/interface/generator/grover_operator.py +34 -23
- classiq/interface/generator/hamiltonian_evolution/exponentiation.py +4 -6
- classiq/interface/generator/hamiltonian_evolution/qdrift.py +4 -4
- classiq/interface/generator/hamiltonian_evolution/suzuki_trotter.py +12 -8
- classiq/interface/generator/hardware/hardware_data.py +76 -36
- classiq/interface/generator/hardware_efficient_ansatz.py +38 -17
- classiq/interface/generator/hartree_fock.py +14 -4
- classiq/interface/generator/identity.py +10 -6
- classiq/interface/generator/linear_pauli_rotations.py +33 -19
- classiq/interface/generator/mcmt_method.py +1 -1
- classiq/interface/generator/mcu.py +20 -16
- classiq/interface/generator/mcx.py +29 -20
- classiq/interface/generator/model/__init__.py +2 -5
- classiq/interface/generator/model/constraints.py +27 -8
- classiq/interface/generator/model/model.py +32 -203
- classiq/interface/generator/model/preferences/preferences.py +118 -43
- classiq/{quantum_register.py → interface/generator/model/quantum_register.py} +27 -22
- classiq/interface/generator/oracles/arithmetic_oracle.py +2 -4
- classiq/interface/generator/oracles/custom_oracle.py +17 -13
- classiq/interface/generator/oracles/oracle_abc.py +9 -9
- classiq/interface/generator/partitioned_register.py +7 -7
- classiq/interface/generator/piecewise_linear_amplitude_loading.py +45 -29
- classiq/interface/generator/preferences/optimization.py +1 -2
- classiq/interface/generator/qpe.py +47 -34
- classiq/interface/generator/qsvm.py +13 -17
- classiq/interface/generator/quantum_function_call.py +107 -87
- classiq/interface/generator/{generated_circuit.py → quantum_program.py} +50 -37
- classiq/interface/generator/range_types.py +13 -12
- classiq/interface/generator/register_role.py +18 -6
- classiq/interface/generator/slice_parsing_utils.py +11 -6
- classiq/interface/generator/standard_gates/controlled_standard_gates.py +32 -39
- classiq/interface/generator/standard_gates/standard_angle_metaclass.py +2 -6
- classiq/interface/generator/standard_gates/standard_gates.py +3 -3
- classiq/interface/generator/standard_gates/u_gate.py +7 -10
- classiq/interface/generator/state_preparation/bell_state_preparation.py +3 -3
- classiq/interface/generator/state_preparation/computational_basis_state_preparation.py +2 -1
- classiq/interface/generator/state_preparation/distributions.py +16 -15
- classiq/interface/generator/state_preparation/metrics.py +5 -7
- classiq/interface/generator/state_preparation/state_preparation.py +30 -23
- classiq/interface/generator/synthesis_metadata/synthesis_duration.py +0 -4
- classiq/interface/generator/synthesis_metadata/synthesis_execution_data.py +20 -6
- classiq/interface/generator/transpiler_basis_gates.py +7 -3
- classiq/interface/generator/types/builtin_enum_declarations.py +178 -0
- classiq/interface/generator/types/compilation_metadata.py +6 -0
- classiq/interface/generator/types/enum_declaration.py +54 -0
- classiq/interface/generator/types/qstruct_declaration.py +18 -0
- classiq/interface/generator/types/struct_declaration.py +15 -14
- classiq/interface/generator/ucc.py +9 -6
- classiq/interface/generator/unitary_gate.py +10 -6
- classiq/interface/generator/user_defined_function_params.py +4 -1
- classiq/interface/generator/validations/flow_graph.py +11 -9
- classiq/interface/generator/validations/validator_functions.py +8 -6
- classiq/interface/generator/visitor.py +23 -16
- classiq/interface/hardware.py +31 -10
- classiq/interface/helpers/classproperty.py +8 -0
- classiq/interface/helpers/custom_encoders.py +3 -0
- classiq/interface/helpers/custom_pydantic_types.py +40 -50
- classiq/interface/helpers/datastructures.py +26 -0
- classiq/interface/helpers/hashable_mixin.py +3 -2
- classiq/interface/helpers/hashable_pydantic_base_model.py +2 -1
- classiq/interface/helpers/pydantic_model_helpers.py +7 -11
- classiq/interface/helpers/validation_helpers.py +4 -21
- classiq/interface/helpers/versioned_model.py +1 -1
- classiq/interface/ide/ide_data.py +16 -20
- classiq/interface/ide/visual_model.py +130 -0
- classiq/interface/interface_version.py +1 -0
- classiq/interface/jobs.py +35 -6
- classiq/interface/model/allocate.py +16 -0
- classiq/interface/model/bind_operation.py +44 -14
- classiq/interface/model/classical_if.py +15 -0
- classiq/interface/model/classical_parameter_declaration.py +33 -3
- classiq/interface/model/control.py +45 -0
- classiq/interface/model/handle_binding.py +298 -20
- classiq/interface/model/inplace_binary_operation.py +31 -26
- classiq/interface/model/invert.py +12 -0
- classiq/interface/model/model.py +87 -73
- classiq/interface/model/native_function_definition.py +16 -21
- classiq/interface/model/parameter.py +13 -0
- classiq/interface/model/phase_operation.py +11 -0
- classiq/interface/model/port_declaration.py +27 -9
- classiq/interface/model/power.py +14 -0
- classiq/interface/model/quantum_expressions/amplitude_loading_operation.py +38 -21
- classiq/interface/model/quantum_expressions/arithmetic_operation.py +51 -14
- classiq/interface/model/quantum_expressions/quantum_expression.py +12 -35
- classiq/interface/model/quantum_function_call.py +146 -462
- classiq/interface/model/quantum_function_declaration.py +193 -152
- classiq/interface/model/quantum_lambda_function.py +65 -0
- classiq/interface/model/quantum_statement.py +71 -12
- classiq/interface/model/quantum_type.py +205 -67
- classiq/interface/model/quantum_variable_declaration.py +4 -26
- classiq/interface/model/repeat.py +15 -0
- classiq/interface/model/statement_block.py +58 -0
- classiq/interface/model/validation_handle.py +13 -6
- classiq/interface/model/variable_declaration_statement.py +3 -1
- classiq/interface/model/within_apply_operation.py +13 -0
- classiq/interface/pyomo_extension/pyomo_sympy_bimap.py +4 -1
- classiq/interface/server/global_versions.py +6 -7
- classiq/interface/server/routes.py +22 -21
- classiq/interface/source_reference.py +59 -0
- classiq/model_expansions/__init__.py +0 -0
- classiq/model_expansions/atomic_expression_functions_defs.py +253 -0
- classiq/model_expansions/capturing/__init__.py +0 -0
- classiq/model_expansions/capturing/captured_vars.py +435 -0
- classiq/model_expansions/capturing/mangling_utils.py +56 -0
- classiq/model_expansions/closure.py +171 -0
- classiq/model_expansions/debug_flag.py +3 -0
- classiq/model_expansions/evaluators/__init__.py +0 -0
- classiq/model_expansions/evaluators/arg_type_match.py +158 -0
- classiq/model_expansions/evaluators/argument_types.py +42 -0
- classiq/model_expansions/evaluators/classical_expression.py +36 -0
- classiq/model_expansions/evaluators/control.py +144 -0
- classiq/model_expansions/evaluators/parameter_types.py +226 -0
- classiq/model_expansions/evaluators/quantum_type_utils.py +239 -0
- classiq/model_expansions/evaluators/type_type_match.py +90 -0
- classiq/model_expansions/expression_evaluator.py +135 -0
- classiq/model_expansions/expression_renamer.py +76 -0
- classiq/model_expansions/function_builder.py +247 -0
- classiq/model_expansions/generative_functions.py +158 -0
- classiq/model_expansions/interpreters/__init__.py +0 -0
- classiq/model_expansions/interpreters/base_interpreter.py +263 -0
- classiq/model_expansions/interpreters/frontend_generative_interpreter.py +28 -0
- classiq/model_expansions/interpreters/generative_interpreter.py +249 -0
- classiq/model_expansions/model_tables.py +18 -0
- classiq/model_expansions/quantum_operations/__init__.py +9 -0
- classiq/model_expansions/quantum_operations/bind.py +60 -0
- classiq/model_expansions/quantum_operations/call_emitter.py +266 -0
- classiq/model_expansions/quantum_operations/classicalif.py +53 -0
- classiq/model_expansions/quantum_operations/declarative_call_emitter.py +87 -0
- classiq/model_expansions/quantum_operations/emitter.py +181 -0
- classiq/model_expansions/quantum_operations/quantum_function_call.py +33 -0
- classiq/model_expansions/quantum_operations/repeat.py +56 -0
- classiq/model_expansions/quantum_operations/shallow_emitter.py +180 -0
- classiq/model_expansions/quantum_operations/variable_decleration.py +28 -0
- classiq/model_expansions/scope.py +240 -0
- classiq/model_expansions/scope_initialization.py +150 -0
- classiq/model_expansions/sympy_conversion/__init__.py +0 -0
- classiq/model_expansions/sympy_conversion/arithmetics.py +49 -0
- classiq/model_expansions/sympy_conversion/expression_to_sympy.py +179 -0
- classiq/model_expansions/sympy_conversion/sympy_to_python.py +123 -0
- classiq/model_expansions/transformers/__init__.py +0 -0
- classiq/model_expansions/transformers/ast_renamer.py +26 -0
- classiq/model_expansions/transformers/var_splitter.py +299 -0
- classiq/model_expansions/utils/__init__.py +0 -0
- classiq/model_expansions/utils/counted_name_allocator.py +11 -0
- classiq/model_expansions/utils/handles_collector.py +33 -0
- classiq/model_expansions/visitors/__init__.py +0 -0
- classiq/model_expansions/visitors/boolean_expression_transformers.py +214 -0
- classiq/model_expansions/visitors/variable_references.py +144 -0
- classiq/open_library/__init__.py +4 -0
- classiq/open_library/functions/__init__.py +130 -0
- classiq/open_library/functions/amplitude_estimation.py +30 -0
- classiq/open_library/functions/discrete_sine_cosine_transform.py +181 -0
- classiq/open_library/functions/grover.py +157 -0
- classiq/open_library/functions/hea.py +115 -0
- classiq/open_library/functions/linear_pauli_rotation.py +82 -0
- classiq/open_library/functions/modular_exponentiation.py +201 -0
- classiq/open_library/functions/qaoa_penalty.py +117 -0
- classiq/open_library/functions/qft_functions.py +54 -0
- classiq/open_library/functions/qpe.py +46 -0
- classiq/open_library/functions/qsvt.py +331 -0
- classiq/open_library/functions/state_preparation.py +301 -0
- classiq/open_library/functions/swap_test.py +27 -0
- classiq/open_library/functions/utility_functions.py +81 -0
- classiq/open_library/functions/variational.py +52 -0
- classiq/qmod/__init__.py +17 -10
- classiq/qmod/builtins/__init__.py +19 -2
- classiq/qmod/builtins/classical_execution_primitives.py +60 -47
- classiq/qmod/builtins/classical_functions.py +44 -38
- classiq/qmod/builtins/constants.py +10 -0
- classiq/qmod/builtins/enums.py +208 -0
- classiq/qmod/builtins/functions/__init__.py +137 -0
- classiq/qmod/builtins/functions/allocation.py +150 -0
- classiq/qmod/builtins/functions/arithmetic.py +55 -0
- classiq/qmod/builtins/functions/benchmarking.py +8 -0
- classiq/qmod/builtins/functions/chemistry.py +91 -0
- classiq/qmod/builtins/functions/exponentiation.py +105 -0
- classiq/qmod/builtins/functions/finance.py +34 -0
- classiq/qmod/builtins/functions/operators.py +16 -0
- classiq/qmod/builtins/functions/qsvm.py +24 -0
- classiq/qmod/builtins/functions/standard_gates.py +651 -0
- classiq/qmod/builtins/operations.py +379 -57
- classiq/qmod/builtins/structs.py +103 -80
- classiq/qmod/cfunc.py +42 -0
- classiq/qmod/classical_function.py +8 -20
- classiq/qmod/cparam.py +64 -0
- classiq/qmod/create_model_function.py +56 -0
- classiq/qmod/declaration_inferrer.py +145 -112
- classiq/qmod/expression_query.py +39 -0
- classiq/qmod/generative.py +42 -0
- classiq/qmod/model_state_container.py +19 -5
- classiq/qmod/native/__init__.py +7 -0
- classiq/qmod/native/expression_to_qmod.py +194 -0
- classiq/qmod/native/pretty_printer.py +401 -0
- classiq/qmod/pretty_print/__init__.py +7 -0
- classiq/qmod/pretty_print/expression_to_python.py +222 -0
- classiq/qmod/pretty_print/pretty_printer.py +572 -0
- classiq/qmod/python_classical_type.py +67 -0
- classiq/qmod/qfunc.py +79 -0
- classiq/qmod/qmod_constant.py +143 -0
- classiq/qmod/qmod_parameter.py +84 -53
- classiq/qmod/qmod_variable.py +497 -100
- classiq/qmod/quantum_callable.py +17 -7
- classiq/qmod/quantum_expandable.py +278 -105
- classiq/qmod/quantum_function.py +232 -48
- classiq/qmod/semantics/__init__.py +0 -0
- classiq/qmod/semantics/annotation/__init__.py +0 -0
- classiq/qmod/semantics/annotation/call_annotation.py +92 -0
- classiq/qmod/semantics/annotation/qstruct_annotator.py +23 -0
- classiq/qmod/semantics/error_manager.py +88 -0
- classiq/qmod/semantics/lambdas.py +25 -0
- classiq/qmod/semantics/static_semantics_visitor.py +384 -0
- classiq/qmod/semantics/validation/__init__.py +0 -0
- classiq/qmod/semantics/validation/constants_validation.py +16 -0
- classiq/qmod/semantics/validation/func_call_validation.py +99 -0
- classiq/qmod/semantics/validation/function_name_collisions_validation.py +23 -0
- classiq/qmod/semantics/validation/handle_validation.py +85 -0
- classiq/qmod/semantics/validation/main_validation.py +33 -0
- classiq/qmod/semantics/validation/types_validation.py +128 -0
- classiq/qmod/symbolic.py +178 -111
- classiq/qmod/symbolic_expr.py +36 -12
- classiq/qmod/symbolic_type.py +2 -5
- classiq/qmod/type_attribute_remover.py +32 -0
- classiq/qmod/utilities.py +108 -1
- classiq/qmod/write_qmod.py +53 -0
- classiq/synthesis.py +210 -22
- {classiq-0.37.1.dist-info → classiq-0.65.3.dist-info}/METADATA +16 -8
- classiq-0.65.3.dist-info/RECORD +521 -0
- {classiq-0.37.1.dist-info → classiq-0.65.3.dist-info}/WHEEL +1 -1
- classiq/_internals/_qfunc_ext.py +0 -6
- classiq/applications/benchmarking/__init__.py +0 -9
- classiq/applications/benchmarking/mirror_benchmarking.py +0 -67
- classiq/applications/numpy_utils.py +0 -37
- classiq/applications_model_constructors/__init__.py +0 -17
- classiq/applications_model_constructors/combinatorial_optimization_model_constructor.py +0 -178
- classiq/applications_model_constructors/grover_model_constructor.py +0 -227
- classiq/applications_model_constructors/libraries/ampltitude_estimation_library.py +0 -11
- classiq/applications_model_constructors/libraries/qmci_library.py +0 -109
- classiq/builtin_functions/__init__.py +0 -43
- classiq/builtin_functions/amplitude_loading.py +0 -3
- classiq/builtin_functions/binary_ops.py +0 -1
- classiq/builtin_functions/exponentiation.py +0 -5
- classiq/builtin_functions/qpe.py +0 -4
- classiq/builtin_functions/qsvm.py +0 -7
- classiq/builtin_functions/range_types.py +0 -5
- classiq/builtin_functions/standard_gates.py +0 -1
- classiq/builtin_functions/state_preparation.py +0 -6
- classiq/builtin_functions/suzuki_trotter.py +0 -3
- classiq/exceptions.py +0 -131
- classiq/interface/executor/aws_execution_cost.py +0 -72
- classiq/interface/executor/error_mitigation.py +0 -6
- classiq/interface/generator/credit_risk_example/linear_gci.py +0 -115
- classiq/interface/generator/credit_risk_example/weighted_adder.py +0 -59
- classiq/interface/generator/expressions/enums/chemistry.py +0 -28
- classiq/interface/generator/expressions/enums/classical_enum.py +0 -5
- classiq/interface/generator/expressions/enums/ladder_operator.py +0 -16
- classiq/interface/generator/expressions/enums/optimizers.py +0 -9
- classiq/interface/generator/expressions/enums/pauli.py +0 -8
- classiq/interface/generator/expressions/enums/qsvm_feature_map_entanglement.py +0 -9
- classiq/interface/generator/expressions/qmod_qnum_proxy.py +0 -22
- classiq/interface/generator/functions/core_lib_declarations/quantum_functions/__init__.py +0 -18
- classiq/interface/generator/functions/core_lib_declarations/quantum_functions/atomic_quantum_functions.py +0 -641
- classiq/interface/generator/functions/core_lib_declarations/quantum_functions/exponentiation_functions.py +0 -89
- classiq/interface/generator/functions/core_lib_declarations/quantum_functions/std_lib_functions.py +0 -862
- classiq/interface/generator/functions/core_lib_declarations/quantum_operators.py +0 -169
- classiq/interface/generator/functions/foreign_function_definition.py +0 -106
- classiq/interface/generator/functions/function_implementation.py +0 -103
- classiq/interface/generator/functions/native_function_definition.py +0 -153
- classiq/interface/generator/functions/quantum_function_declaration.py +0 -69
- classiq/interface/generator/functions/register.py +0 -42
- classiq/interface/generator/functions/register_mapping_data.py +0 -102
- classiq/interface/generator/inequality_mixer.py +0 -51
- classiq/interface/generator/model/classical_main_validator.py +0 -106
- classiq/interface/generator/range_mixer.py +0 -56
- classiq/interface/generator/state_propagator.py +0 -63
- classiq/interface/generator/types/builtin_struct_declarations/__init__.py +0 -2
- classiq/interface/generator/types/builtin_struct_declarations/pauli_struct_declarations.py +0 -22
- classiq/interface/generator/types/builtin_struct_declarations/qaoa_declarations.py +0 -23
- classiq/interface/generator/types/combinatorial_problem.py +0 -26
- classiq/interface/ide/show.py +0 -34
- classiq/interface/model/common_model_types.py +0 -23
- classiq/interface/model/numeric_reinterpretation.py +0 -25
- classiq/interface/model/operator_synthesis_data.py +0 -48
- classiq/interface/model/quantum_expressions/control_state.py +0 -38
- classiq/interface/model/quantum_if_operation.py +0 -95
- classiq/interface/model/resolvers/function_call_resolver.py +0 -43
- classiq/interface/model/validations/handle_validation_base.py +0 -55
- classiq/interface/model/validations/handles_validator.py +0 -154
- classiq/interface/model/validations/port_to_wire_name_generator.py +0 -12
- classiq/model/__init__.py +0 -14
- classiq/model/composite_function_generator.py +0 -33
- classiq/model/function_handler.py +0 -466
- classiq/model/function_handler.pyi +0 -152
- classiq/model/logic_flow.py +0 -149
- classiq/model/logic_flow_change_handler.py +0 -71
- classiq/model/model.py +0 -246
- classiq/qmod/builtins/functions.py +0 -896
- classiq/qmod/qmod_struct.py +0 -37
- classiq/quantum_functions/__init__.py +0 -17
- classiq/quantum_functions/annotation_parser.py +0 -207
- classiq/quantum_functions/decorators.py +0 -22
- classiq/quantum_functions/function_library.py +0 -181
- classiq/quantum_functions/function_parser.py +0 -74
- classiq/quantum_functions/quantum_function.py +0 -236
- classiq-0.37.1.dist-info/RECORD +0 -418
- /classiq/{applications_model_constructors/libraries → applications/combinatorial_helpers}/__init__.py +0 -0
- /classiq/{interface/generator/credit_risk_example → applications/combinatorial_helpers/arithmetic}/__init__.py +0 -0
- /classiq/{interface/generator/functions/core_lib_declarations → applications/combinatorial_helpers/pauli_helpers}/__init__.py +0 -0
- /classiq/{interface/generator/functions/core_lib_declarations/quantum_functions/chemistry_functions.py → applications/combinatorial_helpers/py.typed} +0 -0
- /classiq/{interface/model/resolvers → applications/combinatorial_helpers/solvers}/__init__.py +0 -0
- /classiq/{interface/model/validations → applications/combinatorial_helpers/transformations}/__init__.py +0 -0
- /classiq/{_internals → interface}/enum_utils.py +0 -0
@@ -0,0 +1,42 @@
|
|
1
|
+
from sympy import Eq, Expr, GreaterThan, LessThan, Rational, Symbol, solve
|
2
|
+
from sympy.core.relational import Relational
|
3
|
+
|
4
|
+
from classiq.interface.exceptions import ClassiqCombOptNoSolutionError
|
5
|
+
|
6
|
+
|
7
|
+
def isolate(expression: Expr, var: Symbol) -> Expr:
|
8
|
+
isolated_exp = solve(expression, var)
|
9
|
+
# This converts Rationals to Floats which is needed since we don't support division in arithmetic
|
10
|
+
if isinstance(isolated_exp, Relational) and _has_rational_atom(isolated_exp):
|
11
|
+
isolated_exp = isolated_exp.evalf()
|
12
|
+
|
13
|
+
if isinstance(expression, Eq):
|
14
|
+
if len(isolated_exp) == 1:
|
15
|
+
isolated_exp = type(expression)(var, isolated_exp[0])
|
16
|
+
elif len(isolated_exp) > 1:
|
17
|
+
# sympy didn't manage to isolate an expression
|
18
|
+
isolated_exp = expression
|
19
|
+
else:
|
20
|
+
raise ClassiqCombOptNoSolutionError
|
21
|
+
|
22
|
+
else:
|
23
|
+
if _is_symbol_in_right_hand_side(isolated_exp):
|
24
|
+
isolated_exp = _reverse_ineq_exp(isolated_exp)
|
25
|
+
|
26
|
+
return isolated_exp
|
27
|
+
|
28
|
+
|
29
|
+
def _has_rational_atom(expression: Relational) -> bool:
|
30
|
+
return any(type(atom) is Rational for atom in expression.atoms())
|
31
|
+
|
32
|
+
|
33
|
+
def _is_symbol_in_right_hand_side(expression: Expr) -> bool:
|
34
|
+
return isinstance(expression.args[1], Symbol) and not isinstance(
|
35
|
+
expression.args[0], Symbol
|
36
|
+
)
|
37
|
+
|
38
|
+
|
39
|
+
def _reverse_ineq_exp(expression: Expr) -> Expr:
|
40
|
+
original_relation = type(expression)
|
41
|
+
reverse_relation = LessThan if original_relation is GreaterThan else GreaterThan
|
42
|
+
return reverse_relation(expression.args[1], expression.args[0])
|
@@ -0,0 +1,166 @@
|
|
1
|
+
import itertools
|
2
|
+
from functools import partial
|
3
|
+
from typing import Callable, Union
|
4
|
+
|
5
|
+
import numpy as np
|
6
|
+
import pyomo.environ as pyo
|
7
|
+
|
8
|
+
from classiq.interface.combinatorial_optimization.sense import is_maximization
|
9
|
+
from classiq.interface.combinatorial_optimization.solver_types import QSolver
|
10
|
+
from classiq.interface.exceptions import ClassiqValueError
|
11
|
+
from classiq.interface.executor.vqe_result import VQESolverResult
|
12
|
+
from classiq.interface.generator.functions.qmod_python_interface import QmodPyStruct
|
13
|
+
|
14
|
+
from classiq.applications.combinatorial_helpers.optimization_model import (
|
15
|
+
OptimizationModel,
|
16
|
+
)
|
17
|
+
from classiq.applications.combinatorial_helpers.pauli_helpers.pauli_sparsing import (
|
18
|
+
SparsePauliOp,
|
19
|
+
)
|
20
|
+
from classiq.applications.combinatorial_helpers.pauli_helpers.pauli_utils import (
|
21
|
+
pauli_operator_to_hamiltonian,
|
22
|
+
)
|
23
|
+
from classiq.applications.combinatorial_helpers.pyomo_utils import (
|
24
|
+
add_var_domain_constraints,
|
25
|
+
convert_pyomo_to_global_presentation,
|
26
|
+
evaluate_objective,
|
27
|
+
pyomo_to_qmod_qstruct,
|
28
|
+
)
|
29
|
+
from classiq.qmod.builtins.structs import PauliTerm
|
30
|
+
from classiq.qmod.qmod_variable import QStruct
|
31
|
+
|
32
|
+
|
33
|
+
def compute_qaoa_initial_point(
|
34
|
+
hamiltonian: list[PauliTerm],
|
35
|
+
repetitions: int,
|
36
|
+
) -> list[float]:
|
37
|
+
coeffs_ising = [pauli_term.coefficient for pauli_term in hamiltonian[1:]]
|
38
|
+
# the first coeff is the II...I term
|
39
|
+
coeffs_abs = np.abs(coeffs_ising) # type: ignore[arg-type]
|
40
|
+
coeffs_largest = np.sort(coeffs_abs)[len(coeffs_ising) // 2 :]
|
41
|
+
operator_norm = np.mean(coeffs_largest)
|
42
|
+
time_step = 1 / (2 * operator_norm) # adapted such that for MAXCUT time_step = 1
|
43
|
+
|
44
|
+
beta_params: np.ndarray = np.linspace(1, 0, repetitions) * time_step
|
45
|
+
gamma_params: np.ndarray = np.linspace(0, 1, repetitions) * time_step
|
46
|
+
return list(itertools.chain(*zip(gamma_params, beta_params)))
|
47
|
+
|
48
|
+
|
49
|
+
def pyo_model_to_hamiltonian(
|
50
|
+
pyo_model: pyo.ConcreteModel, penalty_energy: float
|
51
|
+
) -> list[PauliTerm]:
|
52
|
+
pauli_list = OptimizationModel(
|
53
|
+
pyo_model, penalty_energy=penalty_energy, qsolver=QSolver.QAOAPenalty
|
54
|
+
).ising.pauli_list
|
55
|
+
return pauli_operator_to_hamiltonian(pauli_list)
|
56
|
+
|
57
|
+
|
58
|
+
def pyo_model_to_qmod_problem(
|
59
|
+
pyo_model: pyo.ConcreteModel, penalty_energy: float
|
60
|
+
) -> tuple[type[QStruct], Callable]:
|
61
|
+
with add_var_domain_constraints(pyo_model):
|
62
|
+
optimization_model = OptimizationModel(
|
63
|
+
pyo_model, penalty_energy=penalty_energy, qsolver=QSolver.QAOAPenalty
|
64
|
+
)
|
65
|
+
_validate_var_domains(optimization_model)
|
66
|
+
qmod_struct = pyomo_to_qmod_qstruct(
|
67
|
+
"QAOAVars", optimization_model.vars_not_encoded
|
68
|
+
)
|
69
|
+
cost_func = partial(
|
70
|
+
evaluate_objective, *optimization_model.objective_not_encoded_sympy
|
71
|
+
)
|
72
|
+
return qmod_struct, cost_func
|
73
|
+
|
74
|
+
|
75
|
+
def _validate_var_domains(model: OptimizationModel) -> None:
|
76
|
+
for var in model.vars_not_encoded:
|
77
|
+
if (
|
78
|
+
isinstance(var.bounds, tuple)
|
79
|
+
and len(var.bounds) == 2
|
80
|
+
and var.bounds[0] != 0
|
81
|
+
):
|
82
|
+
raise ClassiqValueError(
|
83
|
+
f"Bounds of variable {var.local_name} must start at 0, got {var.bounds}"
|
84
|
+
)
|
85
|
+
|
86
|
+
|
87
|
+
def _str_to_list_int(str_ints: str) -> list[int]:
|
88
|
+
return list(map(int, list(str_ints)))
|
89
|
+
|
90
|
+
|
91
|
+
def _decode_vector_str(
|
92
|
+
optimization_model: OptimizationModel, vector_str: str
|
93
|
+
) -> list[int]:
|
94
|
+
return optimization_model.decode(
|
95
|
+
_str_to_list_int(vector_str[::-1])
|
96
|
+
) # reverse qubit order
|
97
|
+
|
98
|
+
|
99
|
+
def _evaluate_operator(operator: SparsePauliOp, state: Union[list[int], str]) -> float:
|
100
|
+
if isinstance(state, list):
|
101
|
+
state = "".join([str(x) for x in state])
|
102
|
+
|
103
|
+
cost = 0.0
|
104
|
+
for pauli, coeff in zip(operator.paulis, operator.coeffs):
|
105
|
+
expectation = 1
|
106
|
+
for qubit, pauli_char in enumerate(pauli):
|
107
|
+
if pauli_char != "I" and state[qubit] == "1" and pauli_char in ["Z", "Y"]:
|
108
|
+
expectation *= -1
|
109
|
+
cost += expectation * coeff.real
|
110
|
+
|
111
|
+
return cost
|
112
|
+
|
113
|
+
|
114
|
+
def _eigenstate_to_solution(
|
115
|
+
optimization_model: OptimizationModel,
|
116
|
+
eigenvector: str,
|
117
|
+
probability: float,
|
118
|
+
num_shots: int,
|
119
|
+
) -> QmodPyStruct:
|
120
|
+
pauli = optimization_model.get_operator().pauli_list
|
121
|
+
|
122
|
+
paulis = [item[0] for item in pauli]
|
123
|
+
coeffs = [item[1] for item in pauli]
|
124
|
+
|
125
|
+
operator = SparsePauliOp(paulis, coeffs)
|
126
|
+
cost = _evaluate_operator(operator, eigenvector)
|
127
|
+
|
128
|
+
if is_maximization(optimization_model._model_original):
|
129
|
+
cost *= -1
|
130
|
+
|
131
|
+
return {
|
132
|
+
"probability": probability,
|
133
|
+
"cost": cost,
|
134
|
+
"solution": _decode_vector_str(optimization_model, eigenvector),
|
135
|
+
"count": round(probability * num_shots),
|
136
|
+
}
|
137
|
+
|
138
|
+
|
139
|
+
def _get_combi_solution_histogram(
|
140
|
+
optimization_model: OptimizationModel,
|
141
|
+
vqe_result: VQESolverResult,
|
142
|
+
) -> list[QmodPyStruct]:
|
143
|
+
if vqe_result.reduced_probabilities is None:
|
144
|
+
raise ValueError(
|
145
|
+
"reduced_probabilities is optional only for backwards compatibility, but it should always be present here"
|
146
|
+
)
|
147
|
+
return [
|
148
|
+
_eigenstate_to_solution(
|
149
|
+
optimization_model, eigenvector, probability, vqe_result.num_shots
|
150
|
+
)
|
151
|
+
for eigenvector, probability in vqe_result.reduced_probabilities.items()
|
152
|
+
]
|
153
|
+
|
154
|
+
|
155
|
+
def get_optimization_solution_from_pyo(
|
156
|
+
pyo_model: pyo.ConcreteModel,
|
157
|
+
vqe_result: VQESolverResult,
|
158
|
+
penalty_energy: float,
|
159
|
+
) -> list[QmodPyStruct]:
|
160
|
+
converted_pyo_model = convert_pyomo_to_global_presentation(pyo_model)
|
161
|
+
optimization_model = OptimizationModel(
|
162
|
+
converted_pyo_model, penalty_energy=penalty_energy, qsolver=QSolver.QAOAPenalty
|
163
|
+
)
|
164
|
+
histogram = _get_combi_solution_histogram(optimization_model, vqe_result)
|
165
|
+
histogram.sort(key=lambda x: x["probability"], reverse=True)
|
166
|
+
return histogram
|
@@ -0,0 +1,107 @@
|
|
1
|
+
import itertools
|
2
|
+
from dataclasses import dataclass, field
|
3
|
+
from typing import Optional, Union
|
4
|
+
|
5
|
+
import pyomo.environ as pyo
|
6
|
+
from pyomo.core.base import _GeneralVarData
|
7
|
+
from pyomo.core.expr.visitor import clone_expression, identify_variables
|
8
|
+
|
9
|
+
from classiq.interface.combinatorial_optimization.encoding_types import EncodingType
|
10
|
+
from classiq.interface.exceptions import ClassiqCombOptError
|
11
|
+
|
12
|
+
from classiq.applications.combinatorial_helpers import pyomo_utils
|
13
|
+
|
14
|
+
|
15
|
+
@dataclass
|
16
|
+
class VarExpressionMapping:
|
17
|
+
var: _GeneralVarData
|
18
|
+
expr: pyo.Expression
|
19
|
+
encodings_vars: list[_GeneralVarData] = field(default_factory=list)
|
20
|
+
|
21
|
+
|
22
|
+
class EncodingMapping:
|
23
|
+
def __init__(self, encoding_type: EncodingType) -> None:
|
24
|
+
self._data: list[VarExpressionMapping] = []
|
25
|
+
self.encoding_type = encoding_type
|
26
|
+
|
27
|
+
@property
|
28
|
+
def original_vars(self) -> list[_GeneralVarData]:
|
29
|
+
return [pair.var for pair in self._data]
|
30
|
+
|
31
|
+
@property
|
32
|
+
def encodings_vars(self) -> list[_GeneralVarData]:
|
33
|
+
return list(
|
34
|
+
itertools.chain.from_iterable(
|
35
|
+
var_mapping.encodings_vars for var_mapping in self._data
|
36
|
+
)
|
37
|
+
)
|
38
|
+
|
39
|
+
@property
|
40
|
+
def substitution_dict(self) -> dict[int, pyo.Expression]:
|
41
|
+
return {id(mapping.var): mapping.expr for mapping in self._data}
|
42
|
+
|
43
|
+
def __len__(self) -> int:
|
44
|
+
return len(self._data)
|
45
|
+
|
46
|
+
def add(
|
47
|
+
self,
|
48
|
+
original_var: _GeneralVarData,
|
49
|
+
encoding_expr: pyo.Expression,
|
50
|
+
encodings_vars: Union[list[_GeneralVarData], None] = None,
|
51
|
+
) -> None:
|
52
|
+
if encodings_vars is None:
|
53
|
+
encodings_vars = list(identify_variables(encoding_expr))
|
54
|
+
|
55
|
+
self._check_unique_encoding_vars(encodings_vars)
|
56
|
+
self._data.append(
|
57
|
+
VarExpressionMapping(
|
58
|
+
var=original_var, expr=encoding_expr, encodings_vars=encodings_vars
|
59
|
+
)
|
60
|
+
)
|
61
|
+
|
62
|
+
def _check_unique_encoding_vars(self, variables: list[_GeneralVarData]) -> None:
|
63
|
+
assert all(
|
64
|
+
not pyomo_utils.contains(var, self.encodings_vars) for var in variables
|
65
|
+
)
|
66
|
+
|
67
|
+
def get_var_expr_mapping(
|
68
|
+
self, original_var: _GeneralVarData
|
69
|
+
) -> VarExpressionMapping:
|
70
|
+
for var_expr_mapping in self._data:
|
71
|
+
if var_expr_mapping.var is original_var:
|
72
|
+
return var_expr_mapping
|
73
|
+
raise ClassiqCombOptError("No variable expression mapping found.")
|
74
|
+
|
75
|
+
def get_encoding_vars(self, original_var: _GeneralVarData) -> list[_GeneralVarData]:
|
76
|
+
return self.get_var_expr_mapping(original_var).encodings_vars
|
77
|
+
|
78
|
+
def get_original_var(
|
79
|
+
self, encoding_var: _GeneralVarData
|
80
|
+
) -> Optional[_GeneralVarData]:
|
81
|
+
for original_var in self.original_vars:
|
82
|
+
if pyomo_utils.contains(encoding_var, self.get_encoding_vars(original_var)):
|
83
|
+
return original_var
|
84
|
+
return None
|
85
|
+
|
86
|
+
def decode(self, solution: list[int]) -> list[int]:
|
87
|
+
idx = 0
|
88
|
+
decoded_solution = []
|
89
|
+
for var_mapping in self._data:
|
90
|
+
num_encoding_vars = len(var_mapping.encodings_vars)
|
91
|
+
encoding_vars_solution = solution[idx : idx + num_encoding_vars]
|
92
|
+
idx += num_encoding_vars
|
93
|
+
|
94
|
+
substitution_map = {
|
95
|
+
id(var): num
|
96
|
+
for var, num in zip(var_mapping.encodings_vars, encoding_vars_solution)
|
97
|
+
}
|
98
|
+
substituted_expr = clone_expression(
|
99
|
+
var_mapping.expr, substitute=substitution_map
|
100
|
+
)
|
101
|
+
|
102
|
+
if callable(substituted_expr):
|
103
|
+
substituted_expr = substituted_expr()
|
104
|
+
|
105
|
+
decoded_solution.append(substituted_expr)
|
106
|
+
|
107
|
+
return decoded_solution
|
@@ -0,0 +1,124 @@
|
|
1
|
+
import math
|
2
|
+
from itertools import filterfalse
|
3
|
+
from typing import Any, Union, cast
|
4
|
+
|
5
|
+
import numpy as np
|
6
|
+
import pyomo.environ as pyo
|
7
|
+
from pyomo.core.base import _GeneralVarData
|
8
|
+
from pyomo.core.base.component import ComponentData, _ComponentBase
|
9
|
+
from pyomo.core.base.constraint import _GeneralConstraintData
|
10
|
+
from pyomo.core.expr.numeric_expr import (
|
11
|
+
MonomialTermExpression,
|
12
|
+
ProductExpression,
|
13
|
+
clone_expression,
|
14
|
+
)
|
15
|
+
from sympy import Expr
|
16
|
+
|
17
|
+
from classiq.interface.exceptions import ClassiqCombOptNoSolutionError
|
18
|
+
|
19
|
+
from classiq.applications.combinatorial_helpers import pyomo_utils
|
20
|
+
|
21
|
+
_INTEGER_TYPES = [pyo.NonNegativeIntegers, pyo.Integers, pyo.PositiveIntegers]
|
22
|
+
|
23
|
+
|
24
|
+
def is_model_encodable(model: pyo.ConcreteModel) -> bool:
|
25
|
+
variables = pyomo_utils.extract(model, pyo.Var)
|
26
|
+
return not all(is_var_binary(var) for var in variables)
|
27
|
+
|
28
|
+
|
29
|
+
def is_var_binary(var: _GeneralVarData) -> bool:
|
30
|
+
return var.domain == pyo.Binary or (
|
31
|
+
var.domain in _INTEGER_TYPES and var.lb == 0 and var.ub == 1
|
32
|
+
)
|
33
|
+
|
34
|
+
|
35
|
+
def is_var_span_power_of_2(var: _GeneralVarData) -> bool:
|
36
|
+
var_span = get_var_span(var)
|
37
|
+
return math.log2(var_span + 1).is_integer()
|
38
|
+
|
39
|
+
|
40
|
+
ENCODED_SUFFIX = "_encoded"
|
41
|
+
ONE_HOT_SUFFIX = "_one_hot"
|
42
|
+
CONSTRAINT_SUFFIX = "_constraint"
|
43
|
+
|
44
|
+
|
45
|
+
def is_obj_encoded(var: _ComponentBase) -> bool:
|
46
|
+
return ENCODED_SUFFIX in var.name
|
47
|
+
|
48
|
+
|
49
|
+
def get_var_span(var: _GeneralVarData) -> int:
|
50
|
+
return var.ub - var.lb
|
51
|
+
|
52
|
+
|
53
|
+
def encoded_obj_name(name: str) -> str:
|
54
|
+
return name + ENCODED_SUFFIX
|
55
|
+
|
56
|
+
|
57
|
+
def get_encoded_var_index(var: _GeneralVarData) -> int:
|
58
|
+
indexed_var = var.parent_component()
|
59
|
+
index = [
|
60
|
+
index_temp for index_temp, var_temp in indexed_var.items() if var_temp is var
|
61
|
+
][0]
|
62
|
+
return index[1]
|
63
|
+
|
64
|
+
|
65
|
+
def recursively_remove_monomial_expr(obj: Any) -> None:
|
66
|
+
# Due to pyomo bug. see: https://github.com/Pyomo/pyomo/issues/2174
|
67
|
+
for arg in getattr(obj, "args", []):
|
68
|
+
if isinstance(arg, MonomialTermExpression):
|
69
|
+
arg.__class__ = ProductExpression
|
70
|
+
recursively_remove_monomial_expr(arg)
|
71
|
+
|
72
|
+
|
73
|
+
def encode_expr(
|
74
|
+
expr: pyo.Expression, substitution_dict: dict[int, pyo.Expression]
|
75
|
+
) -> pyo.Expression:
|
76
|
+
encoded_expr = clone_expression(expr=expr, substitute=substitution_dict)
|
77
|
+
recursively_remove_monomial_expr(encoded_expr)
|
78
|
+
return encoded_expr
|
79
|
+
|
80
|
+
|
81
|
+
def encode_constraints(
|
82
|
+
model: pyo.ConcreteModel, substitution_dict: dict[int, pyo.Expression]
|
83
|
+
) -> None:
|
84
|
+
all_constraints = pyomo_utils.extract(model, _GeneralConstraintData)
|
85
|
+
constraints = filterfalse(is_obj_encoded, all_constraints)
|
86
|
+
|
87
|
+
for constraint in constraints:
|
88
|
+
constraint_expression = encode_expr(constraint.expr, substitution_dict)
|
89
|
+
if not isinstance(constraint_expression, bool) and not isinstance(
|
90
|
+
constraint_expression, np.bool_
|
91
|
+
):
|
92
|
+
constraint.set_value(expr=constraint_expression)
|
93
|
+
continue
|
94
|
+
|
95
|
+
deal_with_trivial_boolean_constraint(constraint, constraint_expression, model)
|
96
|
+
|
97
|
+
|
98
|
+
def deal_with_trivial_boolean_constraint(
|
99
|
+
constraint: _ComponentBase,
|
100
|
+
constraint_expression: Union[bool, Expr],
|
101
|
+
model: pyo.ConcreteModel,
|
102
|
+
) -> None:
|
103
|
+
# using '==' on purpose since comparing against sympy's True
|
104
|
+
if constraint_expression == True: # noqa: E712
|
105
|
+
pyomo_utils.delete_component(model, cast(ComponentData, constraint))
|
106
|
+
if constraint_expression == False: # noqa: E712
|
107
|
+
raise ClassiqCombOptNoSolutionError
|
108
|
+
|
109
|
+
|
110
|
+
def encode_objective(
|
111
|
+
model: pyo.ConcreteModel, substitution_dict: dict[int, pyo.Expression]
|
112
|
+
) -> None:
|
113
|
+
objective = next(model.component_objects(pyo.Objective))
|
114
|
+
|
115
|
+
encoded_objective = pyo.Objective(
|
116
|
+
expr=encode_expr(objective.expr, substitution_dict),
|
117
|
+
sense=objective.sense,
|
118
|
+
)
|
119
|
+
model.del_component(objective.name)
|
120
|
+
setattr(
|
121
|
+
model,
|
122
|
+
objective.name,
|
123
|
+
encoded_objective,
|
124
|
+
)
|
@@ -0,0 +1,75 @@
|
|
1
|
+
from __future__ import annotations
|
2
|
+
|
3
|
+
from pyomo.core.base import _GeneralVarData
|
4
|
+
|
5
|
+
from classiq.interface.generator.function_params import IOName
|
6
|
+
|
7
|
+
from classiq.applications.combinatorial_helpers import encoding_utils
|
8
|
+
from classiq.applications.combinatorial_helpers.encoding_mapping import EncodingMapping
|
9
|
+
|
10
|
+
AUXILIARY_NAME = "auxiliary"
|
11
|
+
|
12
|
+
|
13
|
+
class InternalQuantumReg:
|
14
|
+
def __init__(self, size: int, name: str) -> None:
|
15
|
+
self.name = name
|
16
|
+
self.size = size
|
17
|
+
|
18
|
+
|
19
|
+
class MemoryMapping:
|
20
|
+
def __init__(
|
21
|
+
self,
|
22
|
+
variables: list[_GeneralVarData],
|
23
|
+
vars_encoding_mapping: EncodingMapping | None = None,
|
24
|
+
) -> None:
|
25
|
+
self.substitution_dict: dict[int, InternalQuantumReg] = dict()
|
26
|
+
self.qubit_allocation: dict[IOName, tuple[int, int]] = dict()
|
27
|
+
self.vars_encoding_mapping: EncodingMapping | None = vars_encoding_mapping
|
28
|
+
self.vars: list[_GeneralVarData] = variables
|
29
|
+
self._allocate_memory()
|
30
|
+
|
31
|
+
def __len__(self) -> int:
|
32
|
+
return len(self.substitution_dict)
|
33
|
+
|
34
|
+
def _allocate_memory(self) -> None:
|
35
|
+
for var_data in self.vars:
|
36
|
+
if (
|
37
|
+
not self.vars_encoding_mapping
|
38
|
+
or encoding_utils.get_var_span(var_data) == 1
|
39
|
+
):
|
40
|
+
num_qubits_var = 1
|
41
|
+
else:
|
42
|
+
num_qubits_var = len(
|
43
|
+
self.vars_encoding_mapping.get_encoding_vars(var_data)
|
44
|
+
)
|
45
|
+
|
46
|
+
self.substitution_dict[id(var_data)] = InternalQuantumReg(
|
47
|
+
num_qubits_var, "q"
|
48
|
+
)
|
49
|
+
var_name = get_var_name(var_data)
|
50
|
+
self.qubit_allocation[var_name] = (
|
51
|
+
self._variable_qubits_allocated,
|
52
|
+
num_qubits_var,
|
53
|
+
)
|
54
|
+
|
55
|
+
@property
|
56
|
+
def qregs(self) -> list[InternalQuantumReg]:
|
57
|
+
return list(self.substitution_dict.values())
|
58
|
+
|
59
|
+
@property
|
60
|
+
def _variable_qubits_allocated(self) -> int:
|
61
|
+
return sum(
|
62
|
+
num_qubits
|
63
|
+
for name, (_, num_qubits) in self.qubit_allocation.items()
|
64
|
+
if name != AUXILIARY_NAME
|
65
|
+
)
|
66
|
+
|
67
|
+
|
68
|
+
def get_var_name(var: _GeneralVarData) -> str:
|
69
|
+
return (
|
70
|
+
var.name.replace("[", "")
|
71
|
+
.replace("]", "")
|
72
|
+
.replace(",", "")
|
73
|
+
.replace("'", "")
|
74
|
+
.replace('"', "")
|
75
|
+
)
|