opentrons 8.6.0__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.
Potentially problematic release.
This version of opentrons might be problematic. Click here for more details.
- opentrons/__init__.py +150 -0
- opentrons/_version.py +34 -0
- opentrons/calibration_storage/__init__.py +54 -0
- opentrons/calibration_storage/deck_configuration.py +62 -0
- opentrons/calibration_storage/encoder_decoder.py +31 -0
- opentrons/calibration_storage/file_operators.py +142 -0
- opentrons/calibration_storage/helpers.py +103 -0
- opentrons/calibration_storage/ot2/__init__.py +34 -0
- opentrons/calibration_storage/ot2/deck_attitude.py +85 -0
- opentrons/calibration_storage/ot2/mark_bad_calibration.py +27 -0
- opentrons/calibration_storage/ot2/models/__init__.py +0 -0
- opentrons/calibration_storage/ot2/models/v1.py +149 -0
- opentrons/calibration_storage/ot2/pipette_offset.py +129 -0
- opentrons/calibration_storage/ot2/tip_length.py +281 -0
- opentrons/calibration_storage/ot3/__init__.py +31 -0
- opentrons/calibration_storage/ot3/deck_attitude.py +83 -0
- opentrons/calibration_storage/ot3/gripper_offset.py +156 -0
- opentrons/calibration_storage/ot3/models/__init__.py +0 -0
- opentrons/calibration_storage/ot3/models/v1.py +122 -0
- opentrons/calibration_storage/ot3/module_offset.py +138 -0
- opentrons/calibration_storage/ot3/pipette_offset.py +95 -0
- opentrons/calibration_storage/types.py +45 -0
- opentrons/cli/__init__.py +21 -0
- opentrons/cli/__main__.py +5 -0
- opentrons/cli/analyze.py +557 -0
- opentrons/config/__init__.py +631 -0
- opentrons/config/advanced_settings.py +871 -0
- opentrons/config/defaults_ot2.py +214 -0
- opentrons/config/defaults_ot3.py +499 -0
- opentrons/config/feature_flags.py +86 -0
- opentrons/config/gripper_config.py +55 -0
- opentrons/config/reset.py +203 -0
- opentrons/config/robot_configs.py +187 -0
- opentrons/config/types.py +183 -0
- opentrons/drivers/__init__.py +0 -0
- opentrons/drivers/absorbance_reader/__init__.py +11 -0
- opentrons/drivers/absorbance_reader/abstract.py +72 -0
- opentrons/drivers/absorbance_reader/async_byonoy.py +352 -0
- opentrons/drivers/absorbance_reader/driver.py +81 -0
- opentrons/drivers/absorbance_reader/hid_protocol.py +161 -0
- opentrons/drivers/absorbance_reader/simulator.py +84 -0
- opentrons/drivers/asyncio/__init__.py +0 -0
- opentrons/drivers/asyncio/communication/__init__.py +22 -0
- opentrons/drivers/asyncio/communication/async_serial.py +187 -0
- opentrons/drivers/asyncio/communication/errors.py +88 -0
- opentrons/drivers/asyncio/communication/serial_connection.py +557 -0
- opentrons/drivers/command_builder.py +102 -0
- opentrons/drivers/flex_stacker/__init__.py +13 -0
- opentrons/drivers/flex_stacker/abstract.py +214 -0
- opentrons/drivers/flex_stacker/driver.py +768 -0
- opentrons/drivers/flex_stacker/errors.py +68 -0
- opentrons/drivers/flex_stacker/simulator.py +309 -0
- opentrons/drivers/flex_stacker/types.py +367 -0
- opentrons/drivers/flex_stacker/utils.py +19 -0
- opentrons/drivers/heater_shaker/__init__.py +5 -0
- opentrons/drivers/heater_shaker/abstract.py +76 -0
- opentrons/drivers/heater_shaker/driver.py +204 -0
- opentrons/drivers/heater_shaker/simulator.py +94 -0
- opentrons/drivers/mag_deck/__init__.py +6 -0
- opentrons/drivers/mag_deck/abstract.py +44 -0
- opentrons/drivers/mag_deck/driver.py +208 -0
- opentrons/drivers/mag_deck/simulator.py +63 -0
- opentrons/drivers/rpi_drivers/__init__.py +33 -0
- opentrons/drivers/rpi_drivers/dev_types.py +94 -0
- opentrons/drivers/rpi_drivers/gpio.py +282 -0
- opentrons/drivers/rpi_drivers/gpio_simulator.py +127 -0
- opentrons/drivers/rpi_drivers/interfaces.py +15 -0
- opentrons/drivers/rpi_drivers/types.py +364 -0
- opentrons/drivers/rpi_drivers/usb.py +102 -0
- opentrons/drivers/rpi_drivers/usb_simulator.py +22 -0
- opentrons/drivers/serial_communication.py +151 -0
- opentrons/drivers/smoothie_drivers/__init__.py +4 -0
- opentrons/drivers/smoothie_drivers/connection.py +51 -0
- opentrons/drivers/smoothie_drivers/constants.py +121 -0
- opentrons/drivers/smoothie_drivers/driver_3_0.py +1933 -0
- opentrons/drivers/smoothie_drivers/errors.py +49 -0
- opentrons/drivers/smoothie_drivers/parse_utils.py +143 -0
- opentrons/drivers/smoothie_drivers/simulator.py +99 -0
- opentrons/drivers/smoothie_drivers/types.py +16 -0
- opentrons/drivers/temp_deck/__init__.py +10 -0
- opentrons/drivers/temp_deck/abstract.py +54 -0
- opentrons/drivers/temp_deck/driver.py +197 -0
- opentrons/drivers/temp_deck/simulator.py +57 -0
- opentrons/drivers/thermocycler/__init__.py +12 -0
- opentrons/drivers/thermocycler/abstract.py +99 -0
- opentrons/drivers/thermocycler/driver.py +395 -0
- opentrons/drivers/thermocycler/simulator.py +126 -0
- opentrons/drivers/types.py +107 -0
- opentrons/drivers/utils.py +222 -0
- opentrons/execute.py +742 -0
- opentrons/hardware_control/__init__.py +65 -0
- opentrons/hardware_control/__main__.py +77 -0
- opentrons/hardware_control/adapters.py +98 -0
- opentrons/hardware_control/api.py +1347 -0
- opentrons/hardware_control/backends/__init__.py +7 -0
- opentrons/hardware_control/backends/controller.py +400 -0
- opentrons/hardware_control/backends/errors.py +9 -0
- opentrons/hardware_control/backends/estop_state.py +164 -0
- opentrons/hardware_control/backends/flex_protocol.py +497 -0
- opentrons/hardware_control/backends/ot3controller.py +1930 -0
- opentrons/hardware_control/backends/ot3simulator.py +900 -0
- opentrons/hardware_control/backends/ot3utils.py +664 -0
- opentrons/hardware_control/backends/simulator.py +442 -0
- opentrons/hardware_control/backends/status_bar_state.py +240 -0
- opentrons/hardware_control/backends/subsystem_manager.py +431 -0
- opentrons/hardware_control/backends/tip_presence_manager.py +173 -0
- opentrons/hardware_control/backends/types.py +14 -0
- opentrons/hardware_control/constants.py +6 -0
- opentrons/hardware_control/dev_types.py +125 -0
- opentrons/hardware_control/emulation/__init__.py +0 -0
- opentrons/hardware_control/emulation/abstract_emulator.py +21 -0
- opentrons/hardware_control/emulation/app.py +56 -0
- opentrons/hardware_control/emulation/connection_handler.py +38 -0
- opentrons/hardware_control/emulation/heater_shaker.py +150 -0
- opentrons/hardware_control/emulation/magdeck.py +60 -0
- opentrons/hardware_control/emulation/module_server/__init__.py +8 -0
- opentrons/hardware_control/emulation/module_server/client.py +78 -0
- opentrons/hardware_control/emulation/module_server/helpers.py +130 -0
- opentrons/hardware_control/emulation/module_server/models.py +31 -0
- opentrons/hardware_control/emulation/module_server/server.py +110 -0
- opentrons/hardware_control/emulation/parser.py +74 -0
- opentrons/hardware_control/emulation/proxy.py +241 -0
- opentrons/hardware_control/emulation/run_emulator.py +68 -0
- opentrons/hardware_control/emulation/scripts/__init__.py +0 -0
- opentrons/hardware_control/emulation/scripts/run_app.py +54 -0
- opentrons/hardware_control/emulation/scripts/run_module_emulator.py +72 -0
- opentrons/hardware_control/emulation/scripts/run_smoothie.py +37 -0
- opentrons/hardware_control/emulation/settings.py +119 -0
- opentrons/hardware_control/emulation/simulations.py +133 -0
- opentrons/hardware_control/emulation/smoothie.py +192 -0
- opentrons/hardware_control/emulation/tempdeck.py +69 -0
- opentrons/hardware_control/emulation/thermocycler.py +128 -0
- opentrons/hardware_control/emulation/types.py +10 -0
- opentrons/hardware_control/emulation/util.py +38 -0
- opentrons/hardware_control/errors.py +43 -0
- opentrons/hardware_control/execution_manager.py +164 -0
- opentrons/hardware_control/instruments/__init__.py +5 -0
- opentrons/hardware_control/instruments/instrument_abc.py +39 -0
- opentrons/hardware_control/instruments/ot2/__init__.py +0 -0
- opentrons/hardware_control/instruments/ot2/instrument_calibration.py +152 -0
- opentrons/hardware_control/instruments/ot2/pipette.py +777 -0
- opentrons/hardware_control/instruments/ot2/pipette_handler.py +995 -0
- opentrons/hardware_control/instruments/ot3/__init__.py +0 -0
- opentrons/hardware_control/instruments/ot3/gripper.py +420 -0
- opentrons/hardware_control/instruments/ot3/gripper_handler.py +173 -0
- opentrons/hardware_control/instruments/ot3/instrument_calibration.py +214 -0
- opentrons/hardware_control/instruments/ot3/pipette.py +858 -0
- opentrons/hardware_control/instruments/ot3/pipette_handler.py +1030 -0
- opentrons/hardware_control/module_control.py +332 -0
- opentrons/hardware_control/modules/__init__.py +69 -0
- opentrons/hardware_control/modules/absorbance_reader.py +373 -0
- opentrons/hardware_control/modules/errors.py +7 -0
- opentrons/hardware_control/modules/flex_stacker.py +948 -0
- opentrons/hardware_control/modules/heater_shaker.py +426 -0
- opentrons/hardware_control/modules/lid_temp_status.py +35 -0
- opentrons/hardware_control/modules/magdeck.py +233 -0
- opentrons/hardware_control/modules/mod_abc.py +245 -0
- opentrons/hardware_control/modules/module_calibration.py +93 -0
- opentrons/hardware_control/modules/plate_temp_status.py +61 -0
- opentrons/hardware_control/modules/tempdeck.py +299 -0
- opentrons/hardware_control/modules/thermocycler.py +731 -0
- opentrons/hardware_control/modules/types.py +417 -0
- opentrons/hardware_control/modules/update.py +255 -0
- opentrons/hardware_control/modules/utils.py +73 -0
- opentrons/hardware_control/motion_utilities.py +318 -0
- opentrons/hardware_control/nozzle_manager.py +422 -0
- opentrons/hardware_control/ot3_calibration.py +1171 -0
- opentrons/hardware_control/ot3api.py +3227 -0
- opentrons/hardware_control/pause_manager.py +31 -0
- opentrons/hardware_control/poller.py +112 -0
- opentrons/hardware_control/protocols/__init__.py +106 -0
- opentrons/hardware_control/protocols/asyncio_configurable.py +11 -0
- opentrons/hardware_control/protocols/calibratable.py +45 -0
- opentrons/hardware_control/protocols/chassis_accessory_manager.py +90 -0
- opentrons/hardware_control/protocols/configurable.py +48 -0
- opentrons/hardware_control/protocols/event_sourcer.py +18 -0
- opentrons/hardware_control/protocols/execution_controllable.py +33 -0
- opentrons/hardware_control/protocols/flex_calibratable.py +96 -0
- opentrons/hardware_control/protocols/flex_instrument_configurer.py +52 -0
- opentrons/hardware_control/protocols/gripper_controller.py +55 -0
- opentrons/hardware_control/protocols/hardware_manager.py +51 -0
- opentrons/hardware_control/protocols/identifiable.py +16 -0
- opentrons/hardware_control/protocols/instrument_configurer.py +206 -0
- opentrons/hardware_control/protocols/liquid_handler.py +266 -0
- opentrons/hardware_control/protocols/module_provider.py +16 -0
- opentrons/hardware_control/protocols/motion_controller.py +243 -0
- opentrons/hardware_control/protocols/position_estimator.py +45 -0
- opentrons/hardware_control/protocols/simulatable.py +10 -0
- opentrons/hardware_control/protocols/stoppable.py +9 -0
- opentrons/hardware_control/protocols/types.py +27 -0
- opentrons/hardware_control/robot_calibration.py +224 -0
- opentrons/hardware_control/scripts/README.md +28 -0
- opentrons/hardware_control/scripts/__init__.py +1 -0
- opentrons/hardware_control/scripts/gripper_control.py +208 -0
- opentrons/hardware_control/scripts/ot3gripper +7 -0
- opentrons/hardware_control/scripts/ot3repl +7 -0
- opentrons/hardware_control/scripts/repl.py +187 -0
- opentrons/hardware_control/scripts/tc_control.py +97 -0
- opentrons/hardware_control/scripts/update_module_fw.py +274 -0
- opentrons/hardware_control/simulator_setup.py +260 -0
- opentrons/hardware_control/thread_manager.py +431 -0
- opentrons/hardware_control/threaded_async_lock.py +97 -0
- opentrons/hardware_control/types.py +792 -0
- opentrons/hardware_control/util.py +234 -0
- opentrons/legacy_broker.py +53 -0
- opentrons/legacy_commands/__init__.py +1 -0
- opentrons/legacy_commands/commands.py +483 -0
- opentrons/legacy_commands/helpers.py +153 -0
- opentrons/legacy_commands/module_commands.py +276 -0
- opentrons/legacy_commands/protocol_commands.py +54 -0
- opentrons/legacy_commands/publisher.py +155 -0
- opentrons/legacy_commands/robot_commands.py +51 -0
- opentrons/legacy_commands/types.py +1186 -0
- opentrons/motion_planning/__init__.py +32 -0
- opentrons/motion_planning/adjacent_slots_getters.py +168 -0
- opentrons/motion_planning/deck_conflict.py +501 -0
- opentrons/motion_planning/errors.py +35 -0
- opentrons/motion_planning/types.py +42 -0
- opentrons/motion_planning/waypoints.py +218 -0
- opentrons/ordered_set.py +138 -0
- opentrons/protocol_api/__init__.py +105 -0
- opentrons/protocol_api/_liquid.py +157 -0
- opentrons/protocol_api/_liquid_properties.py +814 -0
- opentrons/protocol_api/_nozzle_layout.py +31 -0
- opentrons/protocol_api/_parameter_context.py +300 -0
- opentrons/protocol_api/_parameters.py +31 -0
- opentrons/protocol_api/_transfer_liquid_validation.py +108 -0
- opentrons/protocol_api/_types.py +43 -0
- opentrons/protocol_api/config.py +23 -0
- opentrons/protocol_api/core/__init__.py +23 -0
- opentrons/protocol_api/core/common.py +33 -0
- opentrons/protocol_api/core/core_map.py +74 -0
- opentrons/protocol_api/core/engine/__init__.py +22 -0
- opentrons/protocol_api/core/engine/_default_labware_versions.py +179 -0
- opentrons/protocol_api/core/engine/deck_conflict.py +400 -0
- opentrons/protocol_api/core/engine/exceptions.py +19 -0
- opentrons/protocol_api/core/engine/instrument.py +2391 -0
- opentrons/protocol_api/core/engine/labware.py +238 -0
- opentrons/protocol_api/core/engine/load_labware_params.py +73 -0
- opentrons/protocol_api/core/engine/module_core.py +1027 -0
- opentrons/protocol_api/core/engine/overlap_versions.py +20 -0
- opentrons/protocol_api/core/engine/pipette_movement_conflict.py +358 -0
- opentrons/protocol_api/core/engine/point_calculations.py +64 -0
- opentrons/protocol_api/core/engine/protocol.py +1153 -0
- opentrons/protocol_api/core/engine/robot.py +139 -0
- opentrons/protocol_api/core/engine/stringify.py +74 -0
- opentrons/protocol_api/core/engine/transfer_components_executor.py +1006 -0
- opentrons/protocol_api/core/engine/well.py +241 -0
- opentrons/protocol_api/core/instrument.py +459 -0
- opentrons/protocol_api/core/labware.py +151 -0
- opentrons/protocol_api/core/legacy/__init__.py +11 -0
- opentrons/protocol_api/core/legacy/_labware_geometry.py +37 -0
- opentrons/protocol_api/core/legacy/deck.py +369 -0
- opentrons/protocol_api/core/legacy/labware_offset_provider.py +108 -0
- opentrons/protocol_api/core/legacy/legacy_instrument_core.py +709 -0
- opentrons/protocol_api/core/legacy/legacy_labware_core.py +235 -0
- opentrons/protocol_api/core/legacy/legacy_module_core.py +592 -0
- opentrons/protocol_api/core/legacy/legacy_protocol_core.py +612 -0
- opentrons/protocol_api/core/legacy/legacy_well_core.py +162 -0
- opentrons/protocol_api/core/legacy/load_info.py +67 -0
- opentrons/protocol_api/core/legacy/module_geometry.py +547 -0
- opentrons/protocol_api/core/legacy/well_geometry.py +148 -0
- opentrons/protocol_api/core/legacy_simulator/__init__.py +16 -0
- opentrons/protocol_api/core/legacy_simulator/legacy_instrument_core.py +624 -0
- opentrons/protocol_api/core/legacy_simulator/legacy_protocol_core.py +85 -0
- opentrons/protocol_api/core/module.py +484 -0
- opentrons/protocol_api/core/protocol.py +311 -0
- opentrons/protocol_api/core/robot.py +51 -0
- opentrons/protocol_api/core/well.py +116 -0
- opentrons/protocol_api/core/well_grid.py +45 -0
- opentrons/protocol_api/create_protocol_context.py +177 -0
- opentrons/protocol_api/deck.py +223 -0
- opentrons/protocol_api/disposal_locations.py +244 -0
- opentrons/protocol_api/instrument_context.py +3272 -0
- opentrons/protocol_api/labware.py +1579 -0
- opentrons/protocol_api/module_contexts.py +1447 -0
- opentrons/protocol_api/module_validation_and_errors.py +61 -0
- opentrons/protocol_api/protocol_context.py +1688 -0
- opentrons/protocol_api/robot_context.py +303 -0
- opentrons/protocol_api/validation.py +761 -0
- opentrons/protocol_engine/__init__.py +155 -0
- opentrons/protocol_engine/actions/__init__.py +65 -0
- opentrons/protocol_engine/actions/action_dispatcher.py +30 -0
- opentrons/protocol_engine/actions/action_handler.py +13 -0
- opentrons/protocol_engine/actions/actions.py +302 -0
- opentrons/protocol_engine/actions/get_state_update.py +38 -0
- opentrons/protocol_engine/clients/__init__.py +5 -0
- opentrons/protocol_engine/clients/sync_client.py +174 -0
- opentrons/protocol_engine/clients/transports.py +197 -0
- opentrons/protocol_engine/commands/__init__.py +757 -0
- opentrons/protocol_engine/commands/absorbance_reader/__init__.py +61 -0
- opentrons/protocol_engine/commands/absorbance_reader/close_lid.py +154 -0
- opentrons/protocol_engine/commands/absorbance_reader/common.py +6 -0
- opentrons/protocol_engine/commands/absorbance_reader/initialize.py +151 -0
- opentrons/protocol_engine/commands/absorbance_reader/open_lid.py +154 -0
- opentrons/protocol_engine/commands/absorbance_reader/read.py +226 -0
- opentrons/protocol_engine/commands/air_gap_in_place.py +162 -0
- opentrons/protocol_engine/commands/aspirate.py +244 -0
- opentrons/protocol_engine/commands/aspirate_in_place.py +184 -0
- opentrons/protocol_engine/commands/aspirate_while_tracking.py +211 -0
- opentrons/protocol_engine/commands/blow_out.py +146 -0
- opentrons/protocol_engine/commands/blow_out_in_place.py +119 -0
- opentrons/protocol_engine/commands/calibration/__init__.py +60 -0
- opentrons/protocol_engine/commands/calibration/calibrate_gripper.py +166 -0
- opentrons/protocol_engine/commands/calibration/calibrate_module.py +117 -0
- opentrons/protocol_engine/commands/calibration/calibrate_pipette.py +96 -0
- opentrons/protocol_engine/commands/calibration/move_to_maintenance_position.py +156 -0
- opentrons/protocol_engine/commands/command.py +308 -0
- opentrons/protocol_engine/commands/command_unions.py +974 -0
- opentrons/protocol_engine/commands/comment.py +57 -0
- opentrons/protocol_engine/commands/configure_for_volume.py +108 -0
- opentrons/protocol_engine/commands/configure_nozzle_layout.py +115 -0
- opentrons/protocol_engine/commands/custom.py +67 -0
- opentrons/protocol_engine/commands/dispense.py +194 -0
- opentrons/protocol_engine/commands/dispense_in_place.py +179 -0
- opentrons/protocol_engine/commands/dispense_while_tracking.py +204 -0
- opentrons/protocol_engine/commands/drop_tip.py +232 -0
- opentrons/protocol_engine/commands/drop_tip_in_place.py +205 -0
- opentrons/protocol_engine/commands/flex_stacker/__init__.py +64 -0
- opentrons/protocol_engine/commands/flex_stacker/common.py +900 -0
- opentrons/protocol_engine/commands/flex_stacker/empty.py +293 -0
- opentrons/protocol_engine/commands/flex_stacker/fill.py +281 -0
- opentrons/protocol_engine/commands/flex_stacker/retrieve.py +339 -0
- opentrons/protocol_engine/commands/flex_stacker/set_stored_labware.py +328 -0
- opentrons/protocol_engine/commands/flex_stacker/store.py +339 -0
- opentrons/protocol_engine/commands/generate_command_schema.py +61 -0
- opentrons/protocol_engine/commands/get_next_tip.py +134 -0
- opentrons/protocol_engine/commands/get_tip_presence.py +87 -0
- opentrons/protocol_engine/commands/hash_command_params.py +38 -0
- opentrons/protocol_engine/commands/heater_shaker/__init__.py +102 -0
- opentrons/protocol_engine/commands/heater_shaker/close_labware_latch.py +83 -0
- opentrons/protocol_engine/commands/heater_shaker/deactivate_heater.py +82 -0
- opentrons/protocol_engine/commands/heater_shaker/deactivate_shaker.py +84 -0
- opentrons/protocol_engine/commands/heater_shaker/open_labware_latch.py +110 -0
- opentrons/protocol_engine/commands/heater_shaker/set_and_wait_for_shake_speed.py +125 -0
- opentrons/protocol_engine/commands/heater_shaker/set_target_temperature.py +90 -0
- opentrons/protocol_engine/commands/heater_shaker/wait_for_temperature.py +102 -0
- opentrons/protocol_engine/commands/home.py +100 -0
- opentrons/protocol_engine/commands/identify_module.py +86 -0
- opentrons/protocol_engine/commands/labware_handling_common.py +29 -0
- opentrons/protocol_engine/commands/liquid_probe.py +464 -0
- opentrons/protocol_engine/commands/load_labware.py +210 -0
- opentrons/protocol_engine/commands/load_lid.py +154 -0
- opentrons/protocol_engine/commands/load_lid_stack.py +272 -0
- opentrons/protocol_engine/commands/load_liquid.py +95 -0
- opentrons/protocol_engine/commands/load_liquid_class.py +144 -0
- opentrons/protocol_engine/commands/load_module.py +223 -0
- opentrons/protocol_engine/commands/load_pipette.py +167 -0
- opentrons/protocol_engine/commands/magnetic_module/__init__.py +32 -0
- opentrons/protocol_engine/commands/magnetic_module/disengage.py +97 -0
- opentrons/protocol_engine/commands/magnetic_module/engage.py +119 -0
- opentrons/protocol_engine/commands/move_labware.py +546 -0
- opentrons/protocol_engine/commands/move_relative.py +102 -0
- opentrons/protocol_engine/commands/move_to_addressable_area.py +176 -0
- opentrons/protocol_engine/commands/move_to_addressable_area_for_drop_tip.py +198 -0
- opentrons/protocol_engine/commands/move_to_coordinates.py +107 -0
- opentrons/protocol_engine/commands/move_to_well.py +119 -0
- opentrons/protocol_engine/commands/movement_common.py +338 -0
- opentrons/protocol_engine/commands/pick_up_tip.py +241 -0
- opentrons/protocol_engine/commands/pipetting_common.py +443 -0
- opentrons/protocol_engine/commands/prepare_to_aspirate.py +121 -0
- opentrons/protocol_engine/commands/pressure_dispense.py +155 -0
- opentrons/protocol_engine/commands/reload_labware.py +90 -0
- opentrons/protocol_engine/commands/retract_axis.py +75 -0
- opentrons/protocol_engine/commands/robot/__init__.py +70 -0
- opentrons/protocol_engine/commands/robot/close_gripper_jaw.py +96 -0
- opentrons/protocol_engine/commands/robot/common.py +18 -0
- opentrons/protocol_engine/commands/robot/move_axes_relative.py +101 -0
- opentrons/protocol_engine/commands/robot/move_axes_to.py +100 -0
- opentrons/protocol_engine/commands/robot/move_to.py +94 -0
- opentrons/protocol_engine/commands/robot/open_gripper_jaw.py +86 -0
- opentrons/protocol_engine/commands/save_position.py +109 -0
- opentrons/protocol_engine/commands/seal_pipette_to_tip.py +353 -0
- opentrons/protocol_engine/commands/set_rail_lights.py +67 -0
- opentrons/protocol_engine/commands/set_status_bar.py +89 -0
- opentrons/protocol_engine/commands/temperature_module/__init__.py +46 -0
- opentrons/protocol_engine/commands/temperature_module/deactivate.py +86 -0
- opentrons/protocol_engine/commands/temperature_module/set_target_temperature.py +97 -0
- opentrons/protocol_engine/commands/temperature_module/wait_for_temperature.py +104 -0
- opentrons/protocol_engine/commands/thermocycler/__init__.py +152 -0
- opentrons/protocol_engine/commands/thermocycler/close_lid.py +87 -0
- opentrons/protocol_engine/commands/thermocycler/deactivate_block.py +80 -0
- opentrons/protocol_engine/commands/thermocycler/deactivate_lid.py +80 -0
- opentrons/protocol_engine/commands/thermocycler/open_lid.py +87 -0
- opentrons/protocol_engine/commands/thermocycler/run_extended_profile.py +171 -0
- opentrons/protocol_engine/commands/thermocycler/run_profile.py +124 -0
- opentrons/protocol_engine/commands/thermocycler/set_target_block_temperature.py +140 -0
- opentrons/protocol_engine/commands/thermocycler/set_target_lid_temperature.py +100 -0
- opentrons/protocol_engine/commands/thermocycler/wait_for_block_temperature.py +93 -0
- opentrons/protocol_engine/commands/thermocycler/wait_for_lid_temperature.py +89 -0
- opentrons/protocol_engine/commands/touch_tip.py +189 -0
- opentrons/protocol_engine/commands/unsafe/__init__.py +161 -0
- opentrons/protocol_engine/commands/unsafe/unsafe_blow_out_in_place.py +100 -0
- opentrons/protocol_engine/commands/unsafe/unsafe_drop_tip_in_place.py +121 -0
- opentrons/protocol_engine/commands/unsafe/unsafe_engage_axes.py +82 -0
- opentrons/protocol_engine/commands/unsafe/unsafe_place_labware.py +208 -0
- opentrons/protocol_engine/commands/unsafe/unsafe_stacker_close_latch.py +94 -0
- opentrons/protocol_engine/commands/unsafe/unsafe_stacker_manual_retrieve.py +295 -0
- opentrons/protocol_engine/commands/unsafe/unsafe_stacker_open_latch.py +91 -0
- opentrons/protocol_engine/commands/unsafe/unsafe_stacker_prepare_shuttle.py +136 -0
- opentrons/protocol_engine/commands/unsafe/unsafe_ungrip_labware.py +77 -0
- opentrons/protocol_engine/commands/unsafe/update_position_estimators.py +90 -0
- opentrons/protocol_engine/commands/unseal_pipette_from_tip.py +153 -0
- opentrons/protocol_engine/commands/verify_tip_presence.py +100 -0
- opentrons/protocol_engine/commands/wait_for_duration.py +76 -0
- opentrons/protocol_engine/commands/wait_for_resume.py +75 -0
- opentrons/protocol_engine/create_protocol_engine.py +193 -0
- opentrons/protocol_engine/engine_support.py +28 -0
- opentrons/protocol_engine/error_recovery_policy.py +81 -0
- opentrons/protocol_engine/errors/__init__.py +191 -0
- opentrons/protocol_engine/errors/error_occurrence.py +182 -0
- opentrons/protocol_engine/errors/exceptions.py +1308 -0
- opentrons/protocol_engine/execution/__init__.py +50 -0
- opentrons/protocol_engine/execution/command_executor.py +216 -0
- opentrons/protocol_engine/execution/create_queue_worker.py +102 -0
- opentrons/protocol_engine/execution/door_watcher.py +119 -0
- opentrons/protocol_engine/execution/equipment.py +819 -0
- opentrons/protocol_engine/execution/error_recovery_hardware_state_synchronizer.py +101 -0
- opentrons/protocol_engine/execution/gantry_mover.py +686 -0
- opentrons/protocol_engine/execution/hardware_stopper.py +147 -0
- opentrons/protocol_engine/execution/heater_shaker_movement_flagger.py +207 -0
- opentrons/protocol_engine/execution/labware_movement.py +297 -0
- opentrons/protocol_engine/execution/movement.py +350 -0
- opentrons/protocol_engine/execution/pipetting.py +607 -0
- opentrons/protocol_engine/execution/queue_worker.py +86 -0
- opentrons/protocol_engine/execution/rail_lights.py +25 -0
- opentrons/protocol_engine/execution/run_control.py +33 -0
- opentrons/protocol_engine/execution/status_bar.py +34 -0
- opentrons/protocol_engine/execution/thermocycler_movement_flagger.py +188 -0
- opentrons/protocol_engine/execution/thermocycler_plate_lifter.py +81 -0
- opentrons/protocol_engine/execution/tip_handler.py +550 -0
- opentrons/protocol_engine/labware_offset_standardization.py +194 -0
- opentrons/protocol_engine/notes/__init__.py +17 -0
- opentrons/protocol_engine/notes/notes.py +59 -0
- opentrons/protocol_engine/plugins.py +104 -0
- opentrons/protocol_engine/protocol_engine.py +683 -0
- opentrons/protocol_engine/resources/__init__.py +26 -0
- opentrons/protocol_engine/resources/deck_configuration_provider.py +232 -0
- opentrons/protocol_engine/resources/deck_data_provider.py +94 -0
- opentrons/protocol_engine/resources/file_provider.py +161 -0
- opentrons/protocol_engine/resources/fixture_validation.py +68 -0
- opentrons/protocol_engine/resources/labware_data_provider.py +106 -0
- opentrons/protocol_engine/resources/labware_validation.py +73 -0
- opentrons/protocol_engine/resources/model_utils.py +32 -0
- opentrons/protocol_engine/resources/module_data_provider.py +44 -0
- opentrons/protocol_engine/resources/ot3_validation.py +21 -0
- opentrons/protocol_engine/resources/pipette_data_provider.py +379 -0
- opentrons/protocol_engine/slot_standardization.py +128 -0
- opentrons/protocol_engine/state/__init__.py +1 -0
- opentrons/protocol_engine/state/_abstract_store.py +27 -0
- opentrons/protocol_engine/state/_axis_aligned_bounding_box.py +50 -0
- opentrons/protocol_engine/state/_labware_origin_math.py +636 -0
- opentrons/protocol_engine/state/_move_types.py +83 -0
- opentrons/protocol_engine/state/_well_math.py +193 -0
- opentrons/protocol_engine/state/addressable_areas.py +699 -0
- opentrons/protocol_engine/state/command_history.py +309 -0
- opentrons/protocol_engine/state/commands.py +1164 -0
- opentrons/protocol_engine/state/config.py +39 -0
- opentrons/protocol_engine/state/files.py +57 -0
- opentrons/protocol_engine/state/fluid_stack.py +138 -0
- opentrons/protocol_engine/state/geometry.py +2408 -0
- opentrons/protocol_engine/state/inner_well_math_utils.py +548 -0
- opentrons/protocol_engine/state/labware.py +1432 -0
- opentrons/protocol_engine/state/liquid_classes.py +82 -0
- opentrons/protocol_engine/state/liquids.py +73 -0
- opentrons/protocol_engine/state/module_substates/__init__.py +45 -0
- opentrons/protocol_engine/state/module_substates/absorbance_reader_substate.py +35 -0
- opentrons/protocol_engine/state/module_substates/flex_stacker_substate.py +112 -0
- opentrons/protocol_engine/state/module_substates/heater_shaker_module_substate.py +115 -0
- opentrons/protocol_engine/state/module_substates/magnetic_block_substate.py +17 -0
- opentrons/protocol_engine/state/module_substates/magnetic_module_substate.py +65 -0
- opentrons/protocol_engine/state/module_substates/temperature_module_substate.py +67 -0
- opentrons/protocol_engine/state/module_substates/thermocycler_module_substate.py +163 -0
- opentrons/protocol_engine/state/modules.py +1515 -0
- opentrons/protocol_engine/state/motion.py +373 -0
- opentrons/protocol_engine/state/pipettes.py +905 -0
- opentrons/protocol_engine/state/state.py +421 -0
- opentrons/protocol_engine/state/state_summary.py +36 -0
- opentrons/protocol_engine/state/tips.py +420 -0
- opentrons/protocol_engine/state/update_types.py +904 -0
- opentrons/protocol_engine/state/wells.py +290 -0
- opentrons/protocol_engine/types/__init__.py +310 -0
- opentrons/protocol_engine/types/automatic_tip_selection.py +39 -0
- opentrons/protocol_engine/types/command_annotations.py +53 -0
- opentrons/protocol_engine/types/deck_configuration.py +81 -0
- opentrons/protocol_engine/types/execution.py +96 -0
- opentrons/protocol_engine/types/hardware_passthrough.py +25 -0
- opentrons/protocol_engine/types/instrument.py +47 -0
- opentrons/protocol_engine/types/instrument_sensors.py +47 -0
- opentrons/protocol_engine/types/labware.py +131 -0
- opentrons/protocol_engine/types/labware_movement.py +22 -0
- opentrons/protocol_engine/types/labware_offset_location.py +111 -0
- opentrons/protocol_engine/types/labware_offset_vector.py +16 -0
- opentrons/protocol_engine/types/liquid.py +40 -0
- opentrons/protocol_engine/types/liquid_class.py +59 -0
- opentrons/protocol_engine/types/liquid_handling.py +13 -0
- opentrons/protocol_engine/types/liquid_level_detection.py +191 -0
- opentrons/protocol_engine/types/location.py +194 -0
- opentrons/protocol_engine/types/module.py +310 -0
- opentrons/protocol_engine/types/partial_tip_configuration.py +76 -0
- opentrons/protocol_engine/types/run_time_parameters.py +133 -0
- opentrons/protocol_engine/types/tip.py +18 -0
- opentrons/protocol_engine/types/util.py +21 -0
- opentrons/protocol_engine/types/well_position.py +124 -0
- opentrons/protocol_reader/__init__.py +37 -0
- opentrons/protocol_reader/extract_labware_definitions.py +66 -0
- opentrons/protocol_reader/file_format_validator.py +152 -0
- opentrons/protocol_reader/file_hasher.py +27 -0
- opentrons/protocol_reader/file_identifier.py +284 -0
- opentrons/protocol_reader/file_reader_writer.py +90 -0
- opentrons/protocol_reader/input_file.py +16 -0
- opentrons/protocol_reader/protocol_files_invalid_error.py +6 -0
- opentrons/protocol_reader/protocol_reader.py +188 -0
- opentrons/protocol_reader/protocol_source.py +124 -0
- opentrons/protocol_reader/role_analyzer.py +86 -0
- opentrons/protocol_runner/__init__.py +26 -0
- opentrons/protocol_runner/create_simulating_orchestrator.py +118 -0
- opentrons/protocol_runner/json_file_reader.py +55 -0
- opentrons/protocol_runner/json_translator.py +314 -0
- opentrons/protocol_runner/legacy_command_mapper.py +852 -0
- opentrons/protocol_runner/legacy_context_plugin.py +116 -0
- opentrons/protocol_runner/protocol_runner.py +530 -0
- opentrons/protocol_runner/python_protocol_wrappers.py +179 -0
- opentrons/protocol_runner/run_orchestrator.py +496 -0
- opentrons/protocol_runner/task_queue.py +95 -0
- opentrons/protocols/__init__.py +6 -0
- opentrons/protocols/advanced_control/__init__.py +0 -0
- opentrons/protocols/advanced_control/common.py +38 -0
- opentrons/protocols/advanced_control/mix.py +60 -0
- opentrons/protocols/advanced_control/transfers/__init__.py +0 -0
- opentrons/protocols/advanced_control/transfers/common.py +180 -0
- opentrons/protocols/advanced_control/transfers/transfer.py +972 -0
- opentrons/protocols/advanced_control/transfers/transfer_liquid_utils.py +231 -0
- opentrons/protocols/api_support/__init__.py +0 -0
- opentrons/protocols/api_support/constants.py +8 -0
- opentrons/protocols/api_support/deck_type.py +110 -0
- opentrons/protocols/api_support/definitions.py +18 -0
- opentrons/protocols/api_support/instrument.py +151 -0
- opentrons/protocols/api_support/labware_like.py +233 -0
- opentrons/protocols/api_support/tip_tracker.py +175 -0
- opentrons/protocols/api_support/types.py +32 -0
- opentrons/protocols/api_support/util.py +403 -0
- opentrons/protocols/bundle.py +89 -0
- opentrons/protocols/duration/__init__.py +4 -0
- opentrons/protocols/duration/errors.py +5 -0
- opentrons/protocols/duration/estimator.py +628 -0
- opentrons/protocols/execution/__init__.py +0 -0
- opentrons/protocols/execution/dev_types.py +181 -0
- opentrons/protocols/execution/errors.py +40 -0
- opentrons/protocols/execution/execute.py +84 -0
- opentrons/protocols/execution/execute_json_v3.py +275 -0
- opentrons/protocols/execution/execute_json_v4.py +359 -0
- opentrons/protocols/execution/execute_json_v5.py +28 -0
- opentrons/protocols/execution/execute_python.py +169 -0
- opentrons/protocols/execution/json_dispatchers.py +87 -0
- opentrons/protocols/execution/types.py +7 -0
- opentrons/protocols/geometry/__init__.py +0 -0
- opentrons/protocols/geometry/planning.py +297 -0
- opentrons/protocols/labware.py +312 -0
- opentrons/protocols/models/__init__.py +0 -0
- opentrons/protocols/models/json_protocol.py +679 -0
- opentrons/protocols/parameters/__init__.py +0 -0
- opentrons/protocols/parameters/csv_parameter_definition.py +77 -0
- opentrons/protocols/parameters/csv_parameter_interface.py +96 -0
- opentrons/protocols/parameters/exceptions.py +34 -0
- opentrons/protocols/parameters/parameter_definition.py +272 -0
- opentrons/protocols/parameters/types.py +17 -0
- opentrons/protocols/parameters/validation.py +267 -0
- opentrons/protocols/parse.py +671 -0
- opentrons/protocols/types.py +159 -0
- opentrons/py.typed +0 -0
- opentrons/resources/scripts/lpc21isp +0 -0
- opentrons/resources/smoothie-edge-8414642.hex +23010 -0
- opentrons/simulate.py +1065 -0
- opentrons/system/__init__.py +6 -0
- opentrons/system/camera.py +51 -0
- opentrons/system/log_control.py +59 -0
- opentrons/system/nmcli.py +856 -0
- opentrons/system/resin.py +24 -0
- opentrons/system/smoothie_update.py +15 -0
- opentrons/system/wifi.py +204 -0
- opentrons/tools/__init__.py +0 -0
- opentrons/tools/args_handler.py +22 -0
- opentrons/tools/write_pipette_memory.py +157 -0
- opentrons/types.py +618 -0
- opentrons/util/__init__.py +1 -0
- opentrons/util/async_helpers.py +166 -0
- opentrons/util/broker.py +84 -0
- opentrons/util/change_notifier.py +47 -0
- opentrons/util/entrypoint_util.py +278 -0
- opentrons/util/get_union_elements.py +26 -0
- opentrons/util/helpers.py +6 -0
- opentrons/util/linal.py +178 -0
- opentrons/util/logging_config.py +265 -0
- opentrons/util/logging_queue_handler.py +61 -0
- opentrons/util/performance_helpers.py +157 -0
- opentrons-8.6.0.dist-info/METADATA +37 -0
- opentrons-8.6.0.dist-info/RECORD +601 -0
- opentrons-8.6.0.dist-info/WHEEL +4 -0
- opentrons-8.6.0.dist-info/entry_points.txt +3 -0
- opentrons-8.6.0.dist-info/licenses/LICENSE +202 -0
opentrons/execute.py
ADDED
|
@@ -0,0 +1,742 @@
|
|
|
1
|
+
""" opentrons.execute: functions and entrypoint for running protocols
|
|
2
|
+
|
|
3
|
+
This module has functions that can be imported to provide protocol
|
|
4
|
+
contexts for running protocols during interactive sessions like Jupyter or just
|
|
5
|
+
regular python shells. It also provides a console entrypoint for running a
|
|
6
|
+
protocol from the command line.
|
|
7
|
+
"""
|
|
8
|
+
import asyncio
|
|
9
|
+
import atexit
|
|
10
|
+
import argparse
|
|
11
|
+
import contextlib
|
|
12
|
+
import logging
|
|
13
|
+
import os
|
|
14
|
+
import sys
|
|
15
|
+
from typing import (
|
|
16
|
+
TYPE_CHECKING,
|
|
17
|
+
BinaryIO,
|
|
18
|
+
Callable,
|
|
19
|
+
Dict,
|
|
20
|
+
List,
|
|
21
|
+
Optional,
|
|
22
|
+
TextIO,
|
|
23
|
+
Union,
|
|
24
|
+
)
|
|
25
|
+
|
|
26
|
+
from opentrons_shared_data.labware.labware_definition import (
|
|
27
|
+
labware_definition_type_adapter,
|
|
28
|
+
)
|
|
29
|
+
from opentrons_shared_data.robot.types import RobotType
|
|
30
|
+
|
|
31
|
+
from opentrons import protocol_api, __version__, should_use_ot3
|
|
32
|
+
|
|
33
|
+
from opentrons.legacy_commands import types as command_types
|
|
34
|
+
|
|
35
|
+
from opentrons.hardware_control import (
|
|
36
|
+
API as OT2API,
|
|
37
|
+
ThreadManagedHardware,
|
|
38
|
+
ThreadManager,
|
|
39
|
+
)
|
|
40
|
+
from opentrons.hardware_control.types import HardwareFeatureFlags
|
|
41
|
+
|
|
42
|
+
from opentrons.protocols import parse
|
|
43
|
+
from opentrons.protocols.api_support.deck_type import (
|
|
44
|
+
guess_from_global_config as guess_deck_type_from_global_config,
|
|
45
|
+
should_load_fixed_trash,
|
|
46
|
+
should_load_fixed_trash_labware_for_python_protocol,
|
|
47
|
+
)
|
|
48
|
+
from opentrons.protocols.api_support.types import APIVersion
|
|
49
|
+
from opentrons.protocols.execution import execute as execute_apiv2
|
|
50
|
+
from opentrons.protocols.types import (
|
|
51
|
+
ApiDeprecationError,
|
|
52
|
+
Protocol,
|
|
53
|
+
PythonProtocol,
|
|
54
|
+
)
|
|
55
|
+
|
|
56
|
+
from opentrons.protocol_api.core.engine import ENGINE_CORE_API_VERSION
|
|
57
|
+
from opentrons.protocol_api.protocol_context import ProtocolContext
|
|
58
|
+
|
|
59
|
+
from opentrons.protocol_engine import (
|
|
60
|
+
Config,
|
|
61
|
+
DeckType,
|
|
62
|
+
EngineStatus,
|
|
63
|
+
error_recovery_policy,
|
|
64
|
+
)
|
|
65
|
+
from opentrons.protocol_engine.create_protocol_engine import (
|
|
66
|
+
create_protocol_engine_in_thread,
|
|
67
|
+
create_protocol_engine,
|
|
68
|
+
)
|
|
69
|
+
from opentrons.protocol_engine.types import PostRunHardwareState
|
|
70
|
+
|
|
71
|
+
from opentrons.protocol_reader import ProtocolSource
|
|
72
|
+
|
|
73
|
+
from opentrons.protocol_runner import (
|
|
74
|
+
create_protocol_runner,
|
|
75
|
+
RunOrchestrator,
|
|
76
|
+
LiveRunner,
|
|
77
|
+
)
|
|
78
|
+
|
|
79
|
+
from .util import entrypoint_util
|
|
80
|
+
|
|
81
|
+
if TYPE_CHECKING:
|
|
82
|
+
from opentrons_shared_data.labware.types import (
|
|
83
|
+
LabwareDefinition as LabwareDefinitionDict,
|
|
84
|
+
)
|
|
85
|
+
|
|
86
|
+
|
|
87
|
+
_THREAD_MANAGED_HW: Optional[ThreadManagedHardware] = None
|
|
88
|
+
#: The background global cache that all protocol contexts created by
|
|
89
|
+
#: :py:meth:`get_protocol_api` will share
|
|
90
|
+
|
|
91
|
+
|
|
92
|
+
# When a ProtocolContext is using a ProtocolEngine to control the robot, it requires some
|
|
93
|
+
# additional long-lived resources besides _THREAD_MANAGED_HARDWARE. There's a background thread,
|
|
94
|
+
# an asyncio event loop in that thread, and some ProtocolEngine-controlled background tasks in that
|
|
95
|
+
# event loop.
|
|
96
|
+
#
|
|
97
|
+
# When we're executing a protocol file beginning-to-end, we can clean up those resources after it
|
|
98
|
+
# completes. However, when someone gets a live ProtocolContext through get_protocol_api(), we have
|
|
99
|
+
# no way of knowing when they're done with it. So, as a hack, we keep these resources open
|
|
100
|
+
# indefinitely, letting them leak.
|
|
101
|
+
#
|
|
102
|
+
# We keep this at module scope so that the contained context managers aren't garbage-collected.
|
|
103
|
+
# If they're garbage collected, they can close their resources prematurely.
|
|
104
|
+
# https://stackoverflow.com/a/69155026/497934
|
|
105
|
+
_LIVE_PROTOCOL_ENGINE_CONTEXTS = contextlib.ExitStack()
|
|
106
|
+
|
|
107
|
+
|
|
108
|
+
# See Jira RCORE-535.
|
|
109
|
+
_JSON_TOO_NEW_MESSAGE = (
|
|
110
|
+
"Protocols created by recent versions of Protocol Designer"
|
|
111
|
+
" cannot currently be executed with"
|
|
112
|
+
" the opentrons_execute command-line tool"
|
|
113
|
+
" or the opentrons.execute.execute() function."
|
|
114
|
+
" Use the Opentrons App instead."
|
|
115
|
+
)
|
|
116
|
+
|
|
117
|
+
|
|
118
|
+
_EmitRunlogCallable = Callable[[command_types.CommandMessage], None]
|
|
119
|
+
|
|
120
|
+
|
|
121
|
+
def get_protocol_api(
|
|
122
|
+
version: Union[str, APIVersion],
|
|
123
|
+
bundled_labware: Optional[Dict[str, "LabwareDefinitionDict"]] = None,
|
|
124
|
+
bundled_data: Optional[Dict[str, bytes]] = None,
|
|
125
|
+
extra_labware: Optional[Dict[str, "LabwareDefinitionDict"]] = None,
|
|
126
|
+
# If you add any more arguments here, make sure they're kw-only to make mistakes harder in
|
|
127
|
+
# environments without type checking, like Jupyter Notebook.
|
|
128
|
+
# *
|
|
129
|
+
) -> protocol_api.ProtocolContext:
|
|
130
|
+
"""
|
|
131
|
+
Build and return a ``protocol_api.ProtocolContext``
|
|
132
|
+
connected to the robot.
|
|
133
|
+
|
|
134
|
+
This can be used to run protocols from interactive Python sessions
|
|
135
|
+
such as Jupyter or an interpreter on the command line:
|
|
136
|
+
|
|
137
|
+
.. code-block:: python
|
|
138
|
+
|
|
139
|
+
>>> from opentrons.execute import get_protocol_api
|
|
140
|
+
>>> protocol = get_protocol_api('2.0')
|
|
141
|
+
>>> instr = protocol.load_instrument('p300_single', 'right')
|
|
142
|
+
>>> instr.home()
|
|
143
|
+
|
|
144
|
+
When this function is called, modules and instruments will be recached.
|
|
145
|
+
|
|
146
|
+
:param version: The API version to use. This must be lower than
|
|
147
|
+
``opentrons.protocol_api.MAX_SUPPORTED_VERSION``.
|
|
148
|
+
It may be specified either as a string (``'2.0'``) or
|
|
149
|
+
as a ``protocols.types.APIVersion``
|
|
150
|
+
(``APIVersion(2, 0)``).
|
|
151
|
+
:param bundled_labware: If specified, a mapping from labware names to
|
|
152
|
+
labware definitions for labware to consider in the
|
|
153
|
+
protocol. Note that if you specify this, *only*
|
|
154
|
+
labware in this argument will be allowed in the
|
|
155
|
+
protocol. This is preparation for a beta feature
|
|
156
|
+
and is best not used.
|
|
157
|
+
:param bundled_data: If specified, a mapping from filenames to contents
|
|
158
|
+
for data to be available in the protocol from
|
|
159
|
+
:py:obj:`opentrons.protocol_api.ProtocolContext.bundled_data`.
|
|
160
|
+
:param extra_labware: A mapping from labware load names to custom labware definitions.
|
|
161
|
+
If this is ``None`` (the default), and this function is called on a robot,
|
|
162
|
+
it will look for labware in the ``labware`` subdirectory of the Jupyter
|
|
163
|
+
data directory.
|
|
164
|
+
:return: The protocol context.
|
|
165
|
+
"""
|
|
166
|
+
if isinstance(version, str):
|
|
167
|
+
checked_version = parse.version_from_string(version)
|
|
168
|
+
elif not isinstance(version, APIVersion):
|
|
169
|
+
raise TypeError("version must be either a string or an APIVersion")
|
|
170
|
+
else:
|
|
171
|
+
checked_version = version
|
|
172
|
+
|
|
173
|
+
if extra_labware is None:
|
|
174
|
+
extra_labware = {
|
|
175
|
+
uri: details.definition
|
|
176
|
+
for uri, details in (entrypoint_util.find_jupyter_labware() or {}).items()
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
robot_type = _get_robot_type()
|
|
180
|
+
deck_type = guess_deck_type_from_global_config()
|
|
181
|
+
|
|
182
|
+
hardware_controller = _get_global_hardware_controller(robot_type)
|
|
183
|
+
|
|
184
|
+
if checked_version < ENGINE_CORE_API_VERSION:
|
|
185
|
+
context = _create_live_context_non_pe(
|
|
186
|
+
api_version=checked_version,
|
|
187
|
+
deck_type=deck_type,
|
|
188
|
+
hardware_api=hardware_controller,
|
|
189
|
+
bundled_labware=bundled_labware,
|
|
190
|
+
bundled_data=bundled_data,
|
|
191
|
+
extra_labware=extra_labware,
|
|
192
|
+
)
|
|
193
|
+
else:
|
|
194
|
+
if bundled_labware is not None:
|
|
195
|
+
# Protocol Engine has a deep assumption that standard labware definitions are always
|
|
196
|
+
# implicitly loadable.
|
|
197
|
+
raise NotImplementedError(
|
|
198
|
+
f"The bundled_labware argument is not currently supported for Python protocols"
|
|
199
|
+
f" with apiLevel {ENGINE_CORE_API_VERSION} or newer."
|
|
200
|
+
)
|
|
201
|
+
context = _create_live_context_pe(
|
|
202
|
+
api_version=checked_version,
|
|
203
|
+
robot_type=robot_type,
|
|
204
|
+
deck_type=deck_type,
|
|
205
|
+
hardware_api=_THREAD_MANAGED_HW, # type: ignore[arg-type]
|
|
206
|
+
bundled_data=bundled_data,
|
|
207
|
+
extra_labware=extra_labware,
|
|
208
|
+
)
|
|
209
|
+
|
|
210
|
+
hardware_controller.sync.cache_instruments()
|
|
211
|
+
return context
|
|
212
|
+
|
|
213
|
+
|
|
214
|
+
def get_arguments(parser: argparse.ArgumentParser) -> argparse.ArgumentParser:
|
|
215
|
+
"""Get the argument parser for this module
|
|
216
|
+
|
|
217
|
+
Useful if you want to use this module as a component of another CLI program
|
|
218
|
+
and want to add its arguments.
|
|
219
|
+
|
|
220
|
+
:param parser: A parser to add arguments to.
|
|
221
|
+
:returns argparse.ArgumentParser: The parser with arguments added.
|
|
222
|
+
"""
|
|
223
|
+
parser.add_argument(
|
|
224
|
+
"-l",
|
|
225
|
+
"--log-level",
|
|
226
|
+
choices=["debug", "info", "warning", "error", "none"],
|
|
227
|
+
default="warning",
|
|
228
|
+
help="Specify the level filter for logs to show on the command line. "
|
|
229
|
+
"The logs stored in journald or local log files are unaffected by "
|
|
230
|
+
"this option and should be configured in the config file. If "
|
|
231
|
+
"'none', do not show logs",
|
|
232
|
+
)
|
|
233
|
+
parser.add_argument(
|
|
234
|
+
"-L",
|
|
235
|
+
"--custom-labware-path",
|
|
236
|
+
action="append",
|
|
237
|
+
default=[os.getcwd()],
|
|
238
|
+
help="Specify directories to search for custom labware definitions. "
|
|
239
|
+
"You can specify this argument multiple times. Once you specify "
|
|
240
|
+
"a directory in this way, labware definitions in that directory "
|
|
241
|
+
"will become available in ProtocolContext.load_labware(). "
|
|
242
|
+
"Only directories specified directly by "
|
|
243
|
+
"this argument are searched, not their children. JSON files that "
|
|
244
|
+
"do not define labware will be ignored with a message. "
|
|
245
|
+
"The current directory (the one from which you are "
|
|
246
|
+
"invoking this program) will always be included implicitly, "
|
|
247
|
+
"in addition to any directories that you specify.",
|
|
248
|
+
)
|
|
249
|
+
parser.add_argument(
|
|
250
|
+
"-D",
|
|
251
|
+
"--custom-data-path",
|
|
252
|
+
action="append",
|
|
253
|
+
nargs="?",
|
|
254
|
+
const=".",
|
|
255
|
+
default=[],
|
|
256
|
+
help="Specify directories to search for custom data files. "
|
|
257
|
+
"You can specify this argument multiple times. Once you specify "
|
|
258
|
+
"a directory in this way, files located in the specified "
|
|
259
|
+
"directory will be available in ProtocolContext.bundled_data. "
|
|
260
|
+
"Note that bundle execution will still only allow data files in "
|
|
261
|
+
"the bundle. If you specify this without a path, it will "
|
|
262
|
+
"add the current path implicitly. If you do not specify this "
|
|
263
|
+
"argument at all, no data files will be added. Any file in the "
|
|
264
|
+
"specified paths will be loaded into memory and included in the "
|
|
265
|
+
"bundle if --bundle is passed, so be careful that any directory "
|
|
266
|
+
"you specify has only the files you want. It is usually a "
|
|
267
|
+
"better idea to use -d so no files are accidentally included. "
|
|
268
|
+
"Also note that data files are made available as their name, not "
|
|
269
|
+
"their full path, so name them uniquely.",
|
|
270
|
+
)
|
|
271
|
+
parser.add_argument(
|
|
272
|
+
"-d",
|
|
273
|
+
"--custom-data-file",
|
|
274
|
+
action="append",
|
|
275
|
+
default=[],
|
|
276
|
+
help="Specify data files to be made available in "
|
|
277
|
+
"ProtocolContext.bundled_data (and possibly bundled if --bundle "
|
|
278
|
+
"is passed). Can be specified multiple times with different "
|
|
279
|
+
"files. It is usually a better idea to use this than -D because "
|
|
280
|
+
"there is less possibility of accidentally including something.",
|
|
281
|
+
)
|
|
282
|
+
parser.add_argument(
|
|
283
|
+
"protocol",
|
|
284
|
+
metavar="PROTOCOL",
|
|
285
|
+
type=argparse.FileType("rb"),
|
|
286
|
+
help="The protocol file to execute. If you pass '-', you can pipe "
|
|
287
|
+
"the protocol via stdin; this could be useful if you want to use this "
|
|
288
|
+
"utility as part of an automated workflow.",
|
|
289
|
+
)
|
|
290
|
+
return parser
|
|
291
|
+
|
|
292
|
+
|
|
293
|
+
def execute(
|
|
294
|
+
protocol_file: Union[BinaryIO, TextIO],
|
|
295
|
+
protocol_name: str,
|
|
296
|
+
propagate_logs: bool = False,
|
|
297
|
+
log_level: str = "warning",
|
|
298
|
+
emit_runlog: Optional[_EmitRunlogCallable] = None,
|
|
299
|
+
custom_labware_paths: Optional[List[str]] = None,
|
|
300
|
+
custom_data_paths: Optional[List[str]] = None,
|
|
301
|
+
) -> None:
|
|
302
|
+
"""
|
|
303
|
+
Run the protocol itself.
|
|
304
|
+
|
|
305
|
+
This is a one-stop function to run a protocol, whether python or json,
|
|
306
|
+
no matter the api version, from external (i.e. not bound up in other
|
|
307
|
+
internal server infrastructure) sources.
|
|
308
|
+
|
|
309
|
+
To run an opentrons protocol from other places, pass in a file like
|
|
310
|
+
object as protocol_file; this function either returns (if the run has no
|
|
311
|
+
problems) or raises an exception.
|
|
312
|
+
|
|
313
|
+
To call from the command line use either the autogenerated entrypoint
|
|
314
|
+
``opentrons_execute`` or ``python -m opentrons.execute``.
|
|
315
|
+
|
|
316
|
+
:param protocol_file: The protocol file to execute
|
|
317
|
+
:param protocol_name: The name of the protocol file. This is required
|
|
318
|
+
internally, but it may not be a thing we can get
|
|
319
|
+
from the ``protocol_file`` argument.
|
|
320
|
+
:param propagate_logs: Whether this function should allow logs from the
|
|
321
|
+
Opentrons stack to propagate up to the root handler.
|
|
322
|
+
This can be useful if you're integrating this
|
|
323
|
+
function in a larger application, but most logs that
|
|
324
|
+
occur during protocol simulation are best associated
|
|
325
|
+
with the actions in the protocol that cause them.
|
|
326
|
+
Default: ``False``
|
|
327
|
+
:param log_level: The level of logs to emit on the command line:
|
|
328
|
+
``"debug"``, ``"info"``, ``"warning"``, or ``"error"``.
|
|
329
|
+
Defaults to ``"warning"``.
|
|
330
|
+
:param emit_runlog: A callback for printing the run log. If specified, this
|
|
331
|
+
will be called whenever a command adds an entry to the
|
|
332
|
+
run log, which can be used for display and progress
|
|
333
|
+
estimation. If specified, the callback should take a
|
|
334
|
+
single argument (the name doesn't matter) which will
|
|
335
|
+
be a dictionary:
|
|
336
|
+
|
|
337
|
+
.. code-block:: python
|
|
338
|
+
|
|
339
|
+
{
|
|
340
|
+
'name': command_name,
|
|
341
|
+
'payload': {
|
|
342
|
+
'text': string_command_text,
|
|
343
|
+
# The rest of this struct is
|
|
344
|
+
# command-dependent; see
|
|
345
|
+
# opentrons.legacy_commands.commands.
|
|
346
|
+
}
|
|
347
|
+
}
|
|
348
|
+
|
|
349
|
+
.. note::
|
|
350
|
+
In older software versions, ``payload["text"]`` was a
|
|
351
|
+
`format string <https://docs.python.org/3/library/string.html#formatstrings>`_.
|
|
352
|
+
To get human-readable text, you had to do ``payload["text"].format(**payload)``.
|
|
353
|
+
Don't do that anymore. If ``payload["text"]`` happens to contain any
|
|
354
|
+
``{`` or ``}`` characters, it can confuse ``.format()`` and cause it to raise a
|
|
355
|
+
``KeyError``.
|
|
356
|
+
|
|
357
|
+
:param custom_labware_paths: A list of directories to search for custom labware.
|
|
358
|
+
Loads valid labware from these paths and makes them available
|
|
359
|
+
to the protocol context. If this is ``None`` (the default), and
|
|
360
|
+
this function is called on a robot, it will look in the ``labware``
|
|
361
|
+
subdirectory of the Jupyter data directory.
|
|
362
|
+
:param custom_data_paths: A list of directories or files to load custom
|
|
363
|
+
data files from. Ignored if the apiv2 feature
|
|
364
|
+
flag if not set. Entries may be either files or
|
|
365
|
+
directories. Specified files and the
|
|
366
|
+
non-recursive contents of specified directories
|
|
367
|
+
are presented by the protocol context in
|
|
368
|
+
``ProtocolContext.bundled_data``.
|
|
369
|
+
"""
|
|
370
|
+
stack_logger = logging.getLogger("opentrons")
|
|
371
|
+
stack_logger.propagate = propagate_logs
|
|
372
|
+
stack_logger.setLevel(getattr(logging, log_level.upper(), logging.WARNING))
|
|
373
|
+
# TODO(mm, 2023-11-20): We should restore the original log settings when we're done.
|
|
374
|
+
|
|
375
|
+
# TODO(mm, 2023-10-02): Switch this truthy check to `is not None`
|
|
376
|
+
# to match documented behavior.
|
|
377
|
+
# See notes in https://github.com/Opentrons/opentrons/pull/13107
|
|
378
|
+
if custom_labware_paths:
|
|
379
|
+
extra_labware = entrypoint_util.labware_from_paths(custom_labware_paths)
|
|
380
|
+
else:
|
|
381
|
+
extra_labware = entrypoint_util.find_jupyter_labware() or {}
|
|
382
|
+
|
|
383
|
+
if custom_data_paths:
|
|
384
|
+
extra_data = entrypoint_util.datafiles_from_paths(custom_data_paths)
|
|
385
|
+
else:
|
|
386
|
+
extra_data = {}
|
|
387
|
+
|
|
388
|
+
contents = protocol_file.read()
|
|
389
|
+
try:
|
|
390
|
+
protocol = parse.parse(
|
|
391
|
+
contents,
|
|
392
|
+
protocol_name,
|
|
393
|
+
extra_labware={
|
|
394
|
+
uri: details.definition for uri, details in extra_labware.items()
|
|
395
|
+
},
|
|
396
|
+
extra_data=extra_data,
|
|
397
|
+
)
|
|
398
|
+
except parse.JSONSchemaVersionTooNewError as e:
|
|
399
|
+
# opentrons.protocols.parse() doesn't support new JSON protocols.
|
|
400
|
+
# The code to do that should be moved from opentrons.protocol_reader.
|
|
401
|
+
# See https://opentrons.atlassian.net/browse/PLAT-94.
|
|
402
|
+
raise NotImplementedError(_JSON_TOO_NEW_MESSAGE) from e
|
|
403
|
+
|
|
404
|
+
if protocol.api_level < APIVersion(2, 0):
|
|
405
|
+
raise ApiDeprecationError(version=protocol.api_level)
|
|
406
|
+
|
|
407
|
+
# Guard against trying to run protocols for the wrong robot type.
|
|
408
|
+
# This matches what robot-server does.
|
|
409
|
+
# FIXME: This exposes the internal strings "OT-2 Standard" and "OT-3 Standard".
|
|
410
|
+
# https://opentrons.atlassian.net/browse/RSS-370
|
|
411
|
+
if protocol.robot_type != _get_robot_type():
|
|
412
|
+
raise RuntimeError(
|
|
413
|
+
f'This robot is of type "{_get_robot_type()}",'
|
|
414
|
+
f' so it can\'t execute protocols for robot type "{protocol.robot_type}"'
|
|
415
|
+
)
|
|
416
|
+
|
|
417
|
+
if protocol.api_level < ENGINE_CORE_API_VERSION:
|
|
418
|
+
_run_file_non_pe(
|
|
419
|
+
protocol=protocol,
|
|
420
|
+
emit_runlog=emit_runlog,
|
|
421
|
+
)
|
|
422
|
+
else:
|
|
423
|
+
# TODO(mm, 2023-07-06): Once these NotImplementedErrors are resolved, consider removing
|
|
424
|
+
# the enclosing if-else block and running everything through _run_file_pe() for simplicity.
|
|
425
|
+
if custom_data_paths:
|
|
426
|
+
raise NotImplementedError(
|
|
427
|
+
f"The custom_data_paths argument is not currently supported for Python protocols"
|
|
428
|
+
f" with apiLevel {ENGINE_CORE_API_VERSION} or newer."
|
|
429
|
+
)
|
|
430
|
+
protocol_file.seek(0)
|
|
431
|
+
_run_file_pe(
|
|
432
|
+
protocol=protocol,
|
|
433
|
+
hardware_api=_get_global_hardware_controller(_get_robot_type()),
|
|
434
|
+
emit_runlog=emit_runlog,
|
|
435
|
+
)
|
|
436
|
+
|
|
437
|
+
|
|
438
|
+
def make_runlog_cb() -> Callable[[command_types.CommandMessage], None]:
|
|
439
|
+
level = 0
|
|
440
|
+
last_dollar = None
|
|
441
|
+
|
|
442
|
+
def _print_runlog(command: command_types.CommandMessage) -> None:
|
|
443
|
+
nonlocal level
|
|
444
|
+
nonlocal last_dollar
|
|
445
|
+
|
|
446
|
+
if last_dollar == command["$"]:
|
|
447
|
+
if command["$"] == "before":
|
|
448
|
+
level += 1
|
|
449
|
+
else:
|
|
450
|
+
level -= 1
|
|
451
|
+
last_dollar = command["$"]
|
|
452
|
+
if command["$"] == "before":
|
|
453
|
+
print(" ".join(["\t" * level, command["payload"].get("text", "")]))
|
|
454
|
+
|
|
455
|
+
return _print_runlog
|
|
456
|
+
|
|
457
|
+
|
|
458
|
+
def main() -> int:
|
|
459
|
+
"""Handler for command line invocation to run a protocol.
|
|
460
|
+
|
|
461
|
+
:param argv: The arguments the program was invoked with; this is usually
|
|
462
|
+
:py:obj:`sys.argv` but if you want to override that you can.
|
|
463
|
+
:returns int: A success or failure value suitable for use as a shell
|
|
464
|
+
return code passed to :py:obj:`sys.exit` (0 means success,
|
|
465
|
+
anything else is a kind of failure).
|
|
466
|
+
"""
|
|
467
|
+
parser = argparse.ArgumentParser(
|
|
468
|
+
prog="opentrons_execute", description="Run an OT-2 protocol"
|
|
469
|
+
)
|
|
470
|
+
parser = get_arguments(parser)
|
|
471
|
+
# don't want to add this in get_arguments because if somebody upstream is
|
|
472
|
+
# using that parser they probably want their own version
|
|
473
|
+
parser.add_argument("-v", "--version", action="version", version=__version__)
|
|
474
|
+
parser.add_argument(
|
|
475
|
+
"-n",
|
|
476
|
+
"--no-print-runlog",
|
|
477
|
+
action="store_true",
|
|
478
|
+
help="Do not print the commands as they are executed",
|
|
479
|
+
)
|
|
480
|
+
args = parser.parse_args()
|
|
481
|
+
printer = None if args.no_print_runlog else make_runlog_cb()
|
|
482
|
+
if args.log_level != "none":
|
|
483
|
+
stack_logger = logging.getLogger("opentrons")
|
|
484
|
+
stack_logger.addHandler(logging.StreamHandler(sys.stdout))
|
|
485
|
+
log_level = args.log_level
|
|
486
|
+
else:
|
|
487
|
+
# TODO(mm, 2023-07-13): This default logging prints error information redundantly
|
|
488
|
+
# when executing via Protocol Engine, because Protocol Engine logs when commands fail.
|
|
489
|
+
log_level = "warning"
|
|
490
|
+
|
|
491
|
+
try:
|
|
492
|
+
execute(
|
|
493
|
+
protocol_file=args.protocol,
|
|
494
|
+
protocol_name=args.protocol.name,
|
|
495
|
+
custom_labware_paths=args.custom_labware_path,
|
|
496
|
+
custom_data_paths=(args.custom_data_path + args.custom_data_file),
|
|
497
|
+
log_level=log_level,
|
|
498
|
+
emit_runlog=printer,
|
|
499
|
+
)
|
|
500
|
+
return 0
|
|
501
|
+
except entrypoint_util.ProtocolEngineExecuteError as error:
|
|
502
|
+
# This exception is a wrapper that's meaningless to the CLI user.
|
|
503
|
+
# Take the actual protocol problem out of it and just print that.
|
|
504
|
+
print(error.to_stderr_string(), file=sys.stderr)
|
|
505
|
+
return 1
|
|
506
|
+
# execute() might raise other exceptions, but we don't have a nice way to print those.
|
|
507
|
+
# Just let Python show a traceback.
|
|
508
|
+
|
|
509
|
+
|
|
510
|
+
def _create_live_context_non_pe(
|
|
511
|
+
api_version: APIVersion,
|
|
512
|
+
hardware_api: ThreadManagedHardware,
|
|
513
|
+
deck_type: str,
|
|
514
|
+
extra_labware: Optional[Dict[str, "LabwareDefinitionDict"]],
|
|
515
|
+
bundled_labware: Optional[Dict[str, "LabwareDefinitionDict"]],
|
|
516
|
+
bundled_data: Optional[Dict[str, bytes]],
|
|
517
|
+
) -> ProtocolContext:
|
|
518
|
+
"""Return a live ProtocolContext.
|
|
519
|
+
|
|
520
|
+
This controls the robot through the older infrastructure, instead of through Protocol Engine.
|
|
521
|
+
"""
|
|
522
|
+
assert api_version < ENGINE_CORE_API_VERSION
|
|
523
|
+
return protocol_api.create_protocol_context(
|
|
524
|
+
api_version=api_version,
|
|
525
|
+
deck_type=deck_type,
|
|
526
|
+
hardware_api=hardware_api,
|
|
527
|
+
bundled_labware=bundled_labware,
|
|
528
|
+
bundled_data=bundled_data,
|
|
529
|
+
extra_labware=extra_labware,
|
|
530
|
+
)
|
|
531
|
+
|
|
532
|
+
|
|
533
|
+
def _create_live_context_pe(
|
|
534
|
+
api_version: APIVersion,
|
|
535
|
+
hardware_api: ThreadManagedHardware,
|
|
536
|
+
robot_type: RobotType,
|
|
537
|
+
deck_type: str,
|
|
538
|
+
extra_labware: Dict[str, "LabwareDefinitionDict"],
|
|
539
|
+
bundled_data: Optional[Dict[str, bytes]],
|
|
540
|
+
) -> ProtocolContext:
|
|
541
|
+
"""Return a live ProtocolContext that controls the robot through ProtocolEngine."""
|
|
542
|
+
assert api_version >= ENGINE_CORE_API_VERSION
|
|
543
|
+
|
|
544
|
+
global _LIVE_PROTOCOL_ENGINE_CONTEXTS
|
|
545
|
+
hardware_api_wrapped = hardware_api.wrapped()
|
|
546
|
+
pe, loop = _LIVE_PROTOCOL_ENGINE_CONTEXTS.enter_context(
|
|
547
|
+
create_protocol_engine_in_thread(
|
|
548
|
+
hardware_api=hardware_api_wrapped,
|
|
549
|
+
config=_get_protocol_engine_config(),
|
|
550
|
+
deck_configuration=entrypoint_util.get_deck_configuration(),
|
|
551
|
+
file_provider=None,
|
|
552
|
+
error_recovery_policy=error_recovery_policy.never_recover,
|
|
553
|
+
drop_tips_after_run=False,
|
|
554
|
+
post_run_hardware_state=PostRunHardwareState.STAY_ENGAGED_IN_PLACE,
|
|
555
|
+
load_fixed_trash=should_load_fixed_trash_labware_for_python_protocol(
|
|
556
|
+
api_version
|
|
557
|
+
),
|
|
558
|
+
)
|
|
559
|
+
)
|
|
560
|
+
|
|
561
|
+
# `async def` so we can use loop.run_coroutine_threadsafe() to wait for its completion.
|
|
562
|
+
# Non-async would use call_soon_threadsafe(), which makes the waiting harder.
|
|
563
|
+
async def add_all_extra_labware() -> None:
|
|
564
|
+
for labware_definition_dict in extra_labware.values():
|
|
565
|
+
labware_definition = labware_definition_type_adapter.validate_python(
|
|
566
|
+
labware_definition_dict
|
|
567
|
+
)
|
|
568
|
+
pe.add_labware_definition(labware_definition)
|
|
569
|
+
|
|
570
|
+
# Add extra_labware to ProtocolEngine, being careful not to modify ProtocolEngine from this
|
|
571
|
+
# thread. See concurrency notes in ProtocolEngine docstring.
|
|
572
|
+
future = asyncio.run_coroutine_threadsafe(add_all_extra_labware(), loop)
|
|
573
|
+
future.result()
|
|
574
|
+
|
|
575
|
+
return protocol_api.create_protocol_context(
|
|
576
|
+
api_version=api_version,
|
|
577
|
+
hardware_api=hardware_api,
|
|
578
|
+
deck_type=deck_type,
|
|
579
|
+
protocol_engine=pe,
|
|
580
|
+
protocol_engine_loop=loop,
|
|
581
|
+
bundled_data=bundled_data,
|
|
582
|
+
)
|
|
583
|
+
|
|
584
|
+
|
|
585
|
+
def _run_file_non_pe(
|
|
586
|
+
protocol: Protocol,
|
|
587
|
+
emit_runlog: Optional[_EmitRunlogCallable],
|
|
588
|
+
) -> None:
|
|
589
|
+
"""Run a protocol file without Protocol Engine, with the older infrastructure instead."""
|
|
590
|
+
if isinstance(protocol, PythonProtocol):
|
|
591
|
+
extra_labware = protocol.extra_labware
|
|
592
|
+
bundled_labware = protocol.bundled_labware
|
|
593
|
+
bundled_data = protocol.bundled_data
|
|
594
|
+
else:
|
|
595
|
+
# JSON protocols do have "bundled labware" embedded in them, but those aren't represented in
|
|
596
|
+
# the parsed Protocol object and we don't need to create the ProtocolContext with them.
|
|
597
|
+
# execute_apiv2.run_protocol() will pull them out of the JSON and load them into the
|
|
598
|
+
# ProtocolContext.
|
|
599
|
+
extra_labware = None
|
|
600
|
+
bundled_labware = None
|
|
601
|
+
bundled_data = None
|
|
602
|
+
|
|
603
|
+
context = _create_live_context_non_pe(
|
|
604
|
+
api_version=protocol.api_level,
|
|
605
|
+
hardware_api=_get_global_hardware_controller(_get_robot_type()),
|
|
606
|
+
deck_type=guess_deck_type_from_global_config(),
|
|
607
|
+
extra_labware=extra_labware,
|
|
608
|
+
bundled_labware=bundled_labware,
|
|
609
|
+
bundled_data=bundled_data,
|
|
610
|
+
)
|
|
611
|
+
|
|
612
|
+
if emit_runlog:
|
|
613
|
+
context.broker.subscribe(command_types.COMMAND, emit_runlog)
|
|
614
|
+
|
|
615
|
+
context.home()
|
|
616
|
+
try:
|
|
617
|
+
execute_apiv2.run_protocol(
|
|
618
|
+
protocol, context, run_time_parameters_with_overrides=None
|
|
619
|
+
)
|
|
620
|
+
finally:
|
|
621
|
+
context.cleanup()
|
|
622
|
+
|
|
623
|
+
|
|
624
|
+
def _run_file_pe(
|
|
625
|
+
protocol: Protocol,
|
|
626
|
+
hardware_api: ThreadManagedHardware,
|
|
627
|
+
emit_runlog: Optional[_EmitRunlogCallable],
|
|
628
|
+
) -> None:
|
|
629
|
+
"""Run a protocol file with Protocol Engine."""
|
|
630
|
+
|
|
631
|
+
async def run(protocol_source: ProtocolSource) -> None:
|
|
632
|
+
# TODO (spp, 2024-03-18): use run-time param overrides once enabled for cli protocol execution
|
|
633
|
+
hardware_api_wrapped = hardware_api.wrapped()
|
|
634
|
+
protocol_engine = await create_protocol_engine(
|
|
635
|
+
hardware_api=hardware_api_wrapped,
|
|
636
|
+
config=_get_protocol_engine_config(),
|
|
637
|
+
error_recovery_policy=error_recovery_policy.never_recover,
|
|
638
|
+
load_fixed_trash=should_load_fixed_trash(protocol_source.config),
|
|
639
|
+
)
|
|
640
|
+
|
|
641
|
+
protocol_runner = create_protocol_runner(
|
|
642
|
+
protocol_config=protocol_source.config,
|
|
643
|
+
protocol_engine=protocol_engine,
|
|
644
|
+
hardware_api=hardware_api_wrapped,
|
|
645
|
+
)
|
|
646
|
+
|
|
647
|
+
orchestrator = RunOrchestrator(
|
|
648
|
+
hardware_api=hardware_api_wrapped,
|
|
649
|
+
protocol_engine=protocol_engine,
|
|
650
|
+
json_or_python_protocol_runner=protocol_runner,
|
|
651
|
+
fixit_runner=LiveRunner(
|
|
652
|
+
protocol_engine=protocol_engine, hardware_api=hardware_api_wrapped
|
|
653
|
+
),
|
|
654
|
+
setup_runner=LiveRunner(
|
|
655
|
+
protocol_engine=protocol_engine, hardware_api=hardware_api_wrapped
|
|
656
|
+
),
|
|
657
|
+
protocol_live_runner=LiveRunner(
|
|
658
|
+
protocol_engine=protocol_engine, hardware_api=hardware_api_wrapped
|
|
659
|
+
),
|
|
660
|
+
)
|
|
661
|
+
|
|
662
|
+
unsubscribe = protocol_runner.broker.subscribe(
|
|
663
|
+
"command", lambda event: emit_runlog(event) if emit_runlog else None
|
|
664
|
+
)
|
|
665
|
+
try:
|
|
666
|
+
# TODO(mm, 2023-06-30): This will home and drop tips at the end, which is not how
|
|
667
|
+
# things have historically behaved with PAPIv2.13 and older or JSONv5 and older.
|
|
668
|
+
result = await orchestrator.run(
|
|
669
|
+
deck_configuration=entrypoint_util.get_deck_configuration(),
|
|
670
|
+
protocol_source=protocol_source,
|
|
671
|
+
)
|
|
672
|
+
finally:
|
|
673
|
+
unsubscribe()
|
|
674
|
+
|
|
675
|
+
if result.state_summary.status != EngineStatus.SUCCEEDED:
|
|
676
|
+
raise entrypoint_util.ProtocolEngineExecuteError(
|
|
677
|
+
result.state_summary.errors
|
|
678
|
+
)
|
|
679
|
+
|
|
680
|
+
with entrypoint_util.adapt_protocol_source(protocol) as protocol_source:
|
|
681
|
+
asyncio.run(run(protocol_source))
|
|
682
|
+
|
|
683
|
+
|
|
684
|
+
def _get_robot_type() -> RobotType:
|
|
685
|
+
"""Return what kind of robot we're currently running on."""
|
|
686
|
+
return "OT-3 Standard" if should_use_ot3() else "OT-2 Standard"
|
|
687
|
+
|
|
688
|
+
|
|
689
|
+
def _get_protocol_engine_config() -> Config:
|
|
690
|
+
"""Return a Protocol Engine config to execute protocols on this device."""
|
|
691
|
+
return Config(
|
|
692
|
+
robot_type=_get_robot_type(),
|
|
693
|
+
deck_type=DeckType(guess_deck_type_from_global_config()),
|
|
694
|
+
# We deliberately omit ignore_pause=True because, in the current implementation of
|
|
695
|
+
# opentrons.protocol_api.core.engine, that would incorrectly make
|
|
696
|
+
# ProtocolContext.is_simulating() return True.
|
|
697
|
+
)
|
|
698
|
+
|
|
699
|
+
|
|
700
|
+
def _get_global_hardware_controller(robot_type: RobotType) -> ThreadManagedHardware:
|
|
701
|
+
# Build a hardware controller in a worker thread, which is necessary
|
|
702
|
+
# because ipython runs its notebook in asyncio but the notebook
|
|
703
|
+
# is at script/repl scope not function scope and is synchronous so
|
|
704
|
+
# you can't control the loop from inside. If we update to
|
|
705
|
+
# IPython 7 we can avoid this, but for now we can't
|
|
706
|
+
global _THREAD_MANAGED_HW
|
|
707
|
+
if not _THREAD_MANAGED_HW:
|
|
708
|
+
if robot_type == "OT-3 Standard":
|
|
709
|
+
# Conditional import because this isn't installed on OT-2s.
|
|
710
|
+
from opentrons.hardware_control.ot3api import OT3API
|
|
711
|
+
|
|
712
|
+
_THREAD_MANAGED_HW = ThreadManager(
|
|
713
|
+
OT3API.build_hardware_controller,
|
|
714
|
+
feature_flags=HardwareFeatureFlags.build_from_ff(),
|
|
715
|
+
)
|
|
716
|
+
else:
|
|
717
|
+
_THREAD_MANAGED_HW = ThreadManager(
|
|
718
|
+
OT2API.build_hardware_controller,
|
|
719
|
+
feature_flags=HardwareFeatureFlags.build_from_ff(),
|
|
720
|
+
)
|
|
721
|
+
|
|
722
|
+
return _THREAD_MANAGED_HW
|
|
723
|
+
|
|
724
|
+
|
|
725
|
+
@atexit.register
|
|
726
|
+
def _clear_cached_hardware_controller() -> None:
|
|
727
|
+
global _THREAD_MANAGED_HW
|
|
728
|
+
if _THREAD_MANAGED_HW:
|
|
729
|
+
_THREAD_MANAGED_HW.clean_up()
|
|
730
|
+
_THREAD_MANAGED_HW = None
|
|
731
|
+
|
|
732
|
+
|
|
733
|
+
# This atexit registration must come after _clear_cached_hardware_controller()
|
|
734
|
+
# to ensure we tear things down in order from highest level to lowest level.
|
|
735
|
+
@atexit.register
|
|
736
|
+
def _clear_live_protocol_engine_contexts() -> None:
|
|
737
|
+
global _LIVE_PROTOCOL_ENGINE_CONTEXTS
|
|
738
|
+
_LIVE_PROTOCOL_ENGINE_CONTEXTS.close()
|
|
739
|
+
|
|
740
|
+
|
|
741
|
+
if __name__ == "__main__":
|
|
742
|
+
sys.exit(main())
|