opentrons 8.6.0a1__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.

Files changed (600) hide show
  1. opentrons/__init__.py +150 -0
  2. opentrons/_version.py +34 -0
  3. opentrons/calibration_storage/__init__.py +54 -0
  4. opentrons/calibration_storage/deck_configuration.py +62 -0
  5. opentrons/calibration_storage/encoder_decoder.py +31 -0
  6. opentrons/calibration_storage/file_operators.py +142 -0
  7. opentrons/calibration_storage/helpers.py +103 -0
  8. opentrons/calibration_storage/ot2/__init__.py +34 -0
  9. opentrons/calibration_storage/ot2/deck_attitude.py +85 -0
  10. opentrons/calibration_storage/ot2/mark_bad_calibration.py +27 -0
  11. opentrons/calibration_storage/ot2/models/__init__.py +0 -0
  12. opentrons/calibration_storage/ot2/models/v1.py +149 -0
  13. opentrons/calibration_storage/ot2/pipette_offset.py +129 -0
  14. opentrons/calibration_storage/ot2/tip_length.py +281 -0
  15. opentrons/calibration_storage/ot3/__init__.py +31 -0
  16. opentrons/calibration_storage/ot3/deck_attitude.py +83 -0
  17. opentrons/calibration_storage/ot3/gripper_offset.py +156 -0
  18. opentrons/calibration_storage/ot3/models/__init__.py +0 -0
  19. opentrons/calibration_storage/ot3/models/v1.py +122 -0
  20. opentrons/calibration_storage/ot3/module_offset.py +138 -0
  21. opentrons/calibration_storage/ot3/pipette_offset.py +95 -0
  22. opentrons/calibration_storage/types.py +45 -0
  23. opentrons/cli/__init__.py +21 -0
  24. opentrons/cli/__main__.py +5 -0
  25. opentrons/cli/analyze.py +501 -0
  26. opentrons/config/__init__.py +631 -0
  27. opentrons/config/advanced_settings.py +871 -0
  28. opentrons/config/defaults_ot2.py +214 -0
  29. opentrons/config/defaults_ot3.py +499 -0
  30. opentrons/config/feature_flags.py +86 -0
  31. opentrons/config/gripper_config.py +55 -0
  32. opentrons/config/reset.py +203 -0
  33. opentrons/config/robot_configs.py +187 -0
  34. opentrons/config/types.py +183 -0
  35. opentrons/drivers/__init__.py +0 -0
  36. opentrons/drivers/absorbance_reader/__init__.py +11 -0
  37. opentrons/drivers/absorbance_reader/abstract.py +72 -0
  38. opentrons/drivers/absorbance_reader/async_byonoy.py +352 -0
  39. opentrons/drivers/absorbance_reader/driver.py +81 -0
  40. opentrons/drivers/absorbance_reader/hid_protocol.py +161 -0
  41. opentrons/drivers/absorbance_reader/simulator.py +84 -0
  42. opentrons/drivers/asyncio/__init__.py +0 -0
  43. opentrons/drivers/asyncio/communication/__init__.py +22 -0
  44. opentrons/drivers/asyncio/communication/async_serial.py +183 -0
  45. opentrons/drivers/asyncio/communication/errors.py +88 -0
  46. opentrons/drivers/asyncio/communication/serial_connection.py +552 -0
  47. opentrons/drivers/command_builder.py +102 -0
  48. opentrons/drivers/flex_stacker/__init__.py +13 -0
  49. opentrons/drivers/flex_stacker/abstract.py +214 -0
  50. opentrons/drivers/flex_stacker/driver.py +768 -0
  51. opentrons/drivers/flex_stacker/errors.py +68 -0
  52. opentrons/drivers/flex_stacker/simulator.py +309 -0
  53. opentrons/drivers/flex_stacker/types.py +367 -0
  54. opentrons/drivers/flex_stacker/utils.py +19 -0
  55. opentrons/drivers/heater_shaker/__init__.py +5 -0
  56. opentrons/drivers/heater_shaker/abstract.py +76 -0
  57. opentrons/drivers/heater_shaker/driver.py +204 -0
  58. opentrons/drivers/heater_shaker/simulator.py +94 -0
  59. opentrons/drivers/mag_deck/__init__.py +6 -0
  60. opentrons/drivers/mag_deck/abstract.py +44 -0
  61. opentrons/drivers/mag_deck/driver.py +208 -0
  62. opentrons/drivers/mag_deck/simulator.py +63 -0
  63. opentrons/drivers/rpi_drivers/__init__.py +33 -0
  64. opentrons/drivers/rpi_drivers/dev_types.py +94 -0
  65. opentrons/drivers/rpi_drivers/gpio.py +282 -0
  66. opentrons/drivers/rpi_drivers/gpio_simulator.py +127 -0
  67. opentrons/drivers/rpi_drivers/interfaces.py +15 -0
  68. opentrons/drivers/rpi_drivers/types.py +364 -0
  69. opentrons/drivers/rpi_drivers/usb.py +102 -0
  70. opentrons/drivers/rpi_drivers/usb_simulator.py +22 -0
  71. opentrons/drivers/serial_communication.py +151 -0
  72. opentrons/drivers/smoothie_drivers/__init__.py +4 -0
  73. opentrons/drivers/smoothie_drivers/connection.py +51 -0
  74. opentrons/drivers/smoothie_drivers/constants.py +121 -0
  75. opentrons/drivers/smoothie_drivers/driver_3_0.py +1933 -0
  76. opentrons/drivers/smoothie_drivers/errors.py +49 -0
  77. opentrons/drivers/smoothie_drivers/parse_utils.py +143 -0
  78. opentrons/drivers/smoothie_drivers/simulator.py +99 -0
  79. opentrons/drivers/smoothie_drivers/types.py +16 -0
  80. opentrons/drivers/temp_deck/__init__.py +10 -0
  81. opentrons/drivers/temp_deck/abstract.py +54 -0
  82. opentrons/drivers/temp_deck/driver.py +197 -0
  83. opentrons/drivers/temp_deck/simulator.py +57 -0
  84. opentrons/drivers/thermocycler/__init__.py +12 -0
  85. opentrons/drivers/thermocycler/abstract.py +99 -0
  86. opentrons/drivers/thermocycler/driver.py +395 -0
  87. opentrons/drivers/thermocycler/simulator.py +126 -0
  88. opentrons/drivers/types.py +107 -0
  89. opentrons/drivers/utils.py +222 -0
  90. opentrons/execute.py +742 -0
  91. opentrons/hardware_control/__init__.py +65 -0
  92. opentrons/hardware_control/__main__.py +77 -0
  93. opentrons/hardware_control/adapters.py +98 -0
  94. opentrons/hardware_control/api.py +1347 -0
  95. opentrons/hardware_control/backends/__init__.py +7 -0
  96. opentrons/hardware_control/backends/controller.py +400 -0
  97. opentrons/hardware_control/backends/errors.py +9 -0
  98. opentrons/hardware_control/backends/estop_state.py +164 -0
  99. opentrons/hardware_control/backends/flex_protocol.py +497 -0
  100. opentrons/hardware_control/backends/ot3controller.py +1930 -0
  101. opentrons/hardware_control/backends/ot3simulator.py +900 -0
  102. opentrons/hardware_control/backends/ot3utils.py +664 -0
  103. opentrons/hardware_control/backends/simulator.py +442 -0
  104. opentrons/hardware_control/backends/status_bar_state.py +240 -0
  105. opentrons/hardware_control/backends/subsystem_manager.py +431 -0
  106. opentrons/hardware_control/backends/tip_presence_manager.py +173 -0
  107. opentrons/hardware_control/backends/types.py +14 -0
  108. opentrons/hardware_control/constants.py +6 -0
  109. opentrons/hardware_control/dev_types.py +125 -0
  110. opentrons/hardware_control/emulation/__init__.py +0 -0
  111. opentrons/hardware_control/emulation/abstract_emulator.py +21 -0
  112. opentrons/hardware_control/emulation/app.py +56 -0
  113. opentrons/hardware_control/emulation/connection_handler.py +38 -0
  114. opentrons/hardware_control/emulation/heater_shaker.py +150 -0
  115. opentrons/hardware_control/emulation/magdeck.py +60 -0
  116. opentrons/hardware_control/emulation/module_server/__init__.py +8 -0
  117. opentrons/hardware_control/emulation/module_server/client.py +78 -0
  118. opentrons/hardware_control/emulation/module_server/helpers.py +130 -0
  119. opentrons/hardware_control/emulation/module_server/models.py +31 -0
  120. opentrons/hardware_control/emulation/module_server/server.py +110 -0
  121. opentrons/hardware_control/emulation/parser.py +74 -0
  122. opentrons/hardware_control/emulation/proxy.py +241 -0
  123. opentrons/hardware_control/emulation/run_emulator.py +68 -0
  124. opentrons/hardware_control/emulation/scripts/__init__.py +0 -0
  125. opentrons/hardware_control/emulation/scripts/run_app.py +54 -0
  126. opentrons/hardware_control/emulation/scripts/run_module_emulator.py +72 -0
  127. opentrons/hardware_control/emulation/scripts/run_smoothie.py +37 -0
  128. opentrons/hardware_control/emulation/settings.py +119 -0
  129. opentrons/hardware_control/emulation/simulations.py +133 -0
  130. opentrons/hardware_control/emulation/smoothie.py +192 -0
  131. opentrons/hardware_control/emulation/tempdeck.py +69 -0
  132. opentrons/hardware_control/emulation/thermocycler.py +128 -0
  133. opentrons/hardware_control/emulation/types.py +10 -0
  134. opentrons/hardware_control/emulation/util.py +38 -0
  135. opentrons/hardware_control/errors.py +43 -0
  136. opentrons/hardware_control/execution_manager.py +164 -0
  137. opentrons/hardware_control/instruments/__init__.py +5 -0
  138. opentrons/hardware_control/instruments/instrument_abc.py +39 -0
  139. opentrons/hardware_control/instruments/ot2/__init__.py +0 -0
  140. opentrons/hardware_control/instruments/ot2/instrument_calibration.py +152 -0
  141. opentrons/hardware_control/instruments/ot2/pipette.py +777 -0
  142. opentrons/hardware_control/instruments/ot2/pipette_handler.py +995 -0
  143. opentrons/hardware_control/instruments/ot3/__init__.py +0 -0
  144. opentrons/hardware_control/instruments/ot3/gripper.py +420 -0
  145. opentrons/hardware_control/instruments/ot3/gripper_handler.py +173 -0
  146. opentrons/hardware_control/instruments/ot3/instrument_calibration.py +214 -0
  147. opentrons/hardware_control/instruments/ot3/pipette.py +858 -0
  148. opentrons/hardware_control/instruments/ot3/pipette_handler.py +1030 -0
  149. opentrons/hardware_control/module_control.py +332 -0
  150. opentrons/hardware_control/modules/__init__.py +69 -0
  151. opentrons/hardware_control/modules/absorbance_reader.py +373 -0
  152. opentrons/hardware_control/modules/errors.py +7 -0
  153. opentrons/hardware_control/modules/flex_stacker.py +948 -0
  154. opentrons/hardware_control/modules/heater_shaker.py +426 -0
  155. opentrons/hardware_control/modules/lid_temp_status.py +35 -0
  156. opentrons/hardware_control/modules/magdeck.py +233 -0
  157. opentrons/hardware_control/modules/mod_abc.py +245 -0
  158. opentrons/hardware_control/modules/module_calibration.py +93 -0
  159. opentrons/hardware_control/modules/plate_temp_status.py +61 -0
  160. opentrons/hardware_control/modules/tempdeck.py +299 -0
  161. opentrons/hardware_control/modules/thermocycler.py +731 -0
  162. opentrons/hardware_control/modules/types.py +417 -0
  163. opentrons/hardware_control/modules/update.py +255 -0
  164. opentrons/hardware_control/modules/utils.py +73 -0
  165. opentrons/hardware_control/motion_utilities.py +318 -0
  166. opentrons/hardware_control/nozzle_manager.py +422 -0
  167. opentrons/hardware_control/ot3_calibration.py +1171 -0
  168. opentrons/hardware_control/ot3api.py +3227 -0
  169. opentrons/hardware_control/pause_manager.py +31 -0
  170. opentrons/hardware_control/poller.py +112 -0
  171. opentrons/hardware_control/protocols/__init__.py +106 -0
  172. opentrons/hardware_control/protocols/asyncio_configurable.py +11 -0
  173. opentrons/hardware_control/protocols/calibratable.py +45 -0
  174. opentrons/hardware_control/protocols/chassis_accessory_manager.py +90 -0
  175. opentrons/hardware_control/protocols/configurable.py +48 -0
  176. opentrons/hardware_control/protocols/event_sourcer.py +18 -0
  177. opentrons/hardware_control/protocols/execution_controllable.py +33 -0
  178. opentrons/hardware_control/protocols/flex_calibratable.py +96 -0
  179. opentrons/hardware_control/protocols/flex_instrument_configurer.py +52 -0
  180. opentrons/hardware_control/protocols/gripper_controller.py +55 -0
  181. opentrons/hardware_control/protocols/hardware_manager.py +51 -0
  182. opentrons/hardware_control/protocols/identifiable.py +16 -0
  183. opentrons/hardware_control/protocols/instrument_configurer.py +206 -0
  184. opentrons/hardware_control/protocols/liquid_handler.py +266 -0
  185. opentrons/hardware_control/protocols/module_provider.py +16 -0
  186. opentrons/hardware_control/protocols/motion_controller.py +243 -0
  187. opentrons/hardware_control/protocols/position_estimator.py +45 -0
  188. opentrons/hardware_control/protocols/simulatable.py +10 -0
  189. opentrons/hardware_control/protocols/stoppable.py +9 -0
  190. opentrons/hardware_control/protocols/types.py +27 -0
  191. opentrons/hardware_control/robot_calibration.py +224 -0
  192. opentrons/hardware_control/scripts/README.md +28 -0
  193. opentrons/hardware_control/scripts/__init__.py +1 -0
  194. opentrons/hardware_control/scripts/gripper_control.py +208 -0
  195. opentrons/hardware_control/scripts/ot3gripper +7 -0
  196. opentrons/hardware_control/scripts/ot3repl +7 -0
  197. opentrons/hardware_control/scripts/repl.py +187 -0
  198. opentrons/hardware_control/scripts/tc_control.py +97 -0
  199. opentrons/hardware_control/simulator_setup.py +260 -0
  200. opentrons/hardware_control/thread_manager.py +431 -0
  201. opentrons/hardware_control/threaded_async_lock.py +97 -0
  202. opentrons/hardware_control/types.py +792 -0
  203. opentrons/hardware_control/util.py +234 -0
  204. opentrons/legacy_broker.py +53 -0
  205. opentrons/legacy_commands/__init__.py +1 -0
  206. opentrons/legacy_commands/commands.py +483 -0
  207. opentrons/legacy_commands/helpers.py +153 -0
  208. opentrons/legacy_commands/module_commands.py +215 -0
  209. opentrons/legacy_commands/protocol_commands.py +54 -0
  210. opentrons/legacy_commands/publisher.py +155 -0
  211. opentrons/legacy_commands/robot_commands.py +51 -0
  212. opentrons/legacy_commands/types.py +1115 -0
  213. opentrons/motion_planning/__init__.py +32 -0
  214. opentrons/motion_planning/adjacent_slots_getters.py +168 -0
  215. opentrons/motion_planning/deck_conflict.py +396 -0
  216. opentrons/motion_planning/errors.py +35 -0
  217. opentrons/motion_planning/types.py +42 -0
  218. opentrons/motion_planning/waypoints.py +218 -0
  219. opentrons/ordered_set.py +138 -0
  220. opentrons/protocol_api/__init__.py +105 -0
  221. opentrons/protocol_api/_liquid.py +157 -0
  222. opentrons/protocol_api/_liquid_properties.py +814 -0
  223. opentrons/protocol_api/_nozzle_layout.py +31 -0
  224. opentrons/protocol_api/_parameter_context.py +300 -0
  225. opentrons/protocol_api/_parameters.py +31 -0
  226. opentrons/protocol_api/_transfer_liquid_validation.py +108 -0
  227. opentrons/protocol_api/_types.py +43 -0
  228. opentrons/protocol_api/config.py +23 -0
  229. opentrons/protocol_api/core/__init__.py +23 -0
  230. opentrons/protocol_api/core/common.py +33 -0
  231. opentrons/protocol_api/core/core_map.py +74 -0
  232. opentrons/protocol_api/core/engine/__init__.py +22 -0
  233. opentrons/protocol_api/core/engine/_default_labware_versions.py +179 -0
  234. opentrons/protocol_api/core/engine/deck_conflict.py +348 -0
  235. opentrons/protocol_api/core/engine/exceptions.py +19 -0
  236. opentrons/protocol_api/core/engine/instrument.py +2391 -0
  237. opentrons/protocol_api/core/engine/labware.py +238 -0
  238. opentrons/protocol_api/core/engine/load_labware_params.py +73 -0
  239. opentrons/protocol_api/core/engine/module_core.py +1025 -0
  240. opentrons/protocol_api/core/engine/overlap_versions.py +20 -0
  241. opentrons/protocol_api/core/engine/pipette_movement_conflict.py +358 -0
  242. opentrons/protocol_api/core/engine/point_calculations.py +64 -0
  243. opentrons/protocol_api/core/engine/protocol.py +1153 -0
  244. opentrons/protocol_api/core/engine/robot.py +139 -0
  245. opentrons/protocol_api/core/engine/stringify.py +74 -0
  246. opentrons/protocol_api/core/engine/transfer_components_executor.py +990 -0
  247. opentrons/protocol_api/core/engine/well.py +241 -0
  248. opentrons/protocol_api/core/instrument.py +459 -0
  249. opentrons/protocol_api/core/labware.py +151 -0
  250. opentrons/protocol_api/core/legacy/__init__.py +11 -0
  251. opentrons/protocol_api/core/legacy/_labware_geometry.py +37 -0
  252. opentrons/protocol_api/core/legacy/deck.py +369 -0
  253. opentrons/protocol_api/core/legacy/labware_offset_provider.py +108 -0
  254. opentrons/protocol_api/core/legacy/legacy_instrument_core.py +709 -0
  255. opentrons/protocol_api/core/legacy/legacy_labware_core.py +235 -0
  256. opentrons/protocol_api/core/legacy/legacy_module_core.py +592 -0
  257. opentrons/protocol_api/core/legacy/legacy_protocol_core.py +612 -0
  258. opentrons/protocol_api/core/legacy/legacy_well_core.py +162 -0
  259. opentrons/protocol_api/core/legacy/load_info.py +67 -0
  260. opentrons/protocol_api/core/legacy/module_geometry.py +547 -0
  261. opentrons/protocol_api/core/legacy/well_geometry.py +148 -0
  262. opentrons/protocol_api/core/legacy_simulator/__init__.py +16 -0
  263. opentrons/protocol_api/core/legacy_simulator/legacy_instrument_core.py +624 -0
  264. opentrons/protocol_api/core/legacy_simulator/legacy_protocol_core.py +85 -0
  265. opentrons/protocol_api/core/module.py +484 -0
  266. opentrons/protocol_api/core/protocol.py +311 -0
  267. opentrons/protocol_api/core/robot.py +51 -0
  268. opentrons/protocol_api/core/well.py +116 -0
  269. opentrons/protocol_api/core/well_grid.py +45 -0
  270. opentrons/protocol_api/create_protocol_context.py +177 -0
  271. opentrons/protocol_api/deck.py +223 -0
  272. opentrons/protocol_api/disposal_locations.py +244 -0
  273. opentrons/protocol_api/instrument_context.py +3212 -0
  274. opentrons/protocol_api/labware.py +1579 -0
  275. opentrons/protocol_api/module_contexts.py +1425 -0
  276. opentrons/protocol_api/module_validation_and_errors.py +61 -0
  277. opentrons/protocol_api/protocol_context.py +1688 -0
  278. opentrons/protocol_api/robot_context.py +303 -0
  279. opentrons/protocol_api/validation.py +761 -0
  280. opentrons/protocol_engine/__init__.py +155 -0
  281. opentrons/protocol_engine/actions/__init__.py +65 -0
  282. opentrons/protocol_engine/actions/action_dispatcher.py +30 -0
  283. opentrons/protocol_engine/actions/action_handler.py +13 -0
  284. opentrons/protocol_engine/actions/actions.py +302 -0
  285. opentrons/protocol_engine/actions/get_state_update.py +38 -0
  286. opentrons/protocol_engine/clients/__init__.py +5 -0
  287. opentrons/protocol_engine/clients/sync_client.py +174 -0
  288. opentrons/protocol_engine/clients/transports.py +197 -0
  289. opentrons/protocol_engine/commands/__init__.py +757 -0
  290. opentrons/protocol_engine/commands/absorbance_reader/__init__.py +61 -0
  291. opentrons/protocol_engine/commands/absorbance_reader/close_lid.py +154 -0
  292. opentrons/protocol_engine/commands/absorbance_reader/common.py +6 -0
  293. opentrons/protocol_engine/commands/absorbance_reader/initialize.py +151 -0
  294. opentrons/protocol_engine/commands/absorbance_reader/open_lid.py +154 -0
  295. opentrons/protocol_engine/commands/absorbance_reader/read.py +226 -0
  296. opentrons/protocol_engine/commands/air_gap_in_place.py +162 -0
  297. opentrons/protocol_engine/commands/aspirate.py +244 -0
  298. opentrons/protocol_engine/commands/aspirate_in_place.py +184 -0
  299. opentrons/protocol_engine/commands/aspirate_while_tracking.py +211 -0
  300. opentrons/protocol_engine/commands/blow_out.py +146 -0
  301. opentrons/protocol_engine/commands/blow_out_in_place.py +119 -0
  302. opentrons/protocol_engine/commands/calibration/__init__.py +60 -0
  303. opentrons/protocol_engine/commands/calibration/calibrate_gripper.py +166 -0
  304. opentrons/protocol_engine/commands/calibration/calibrate_module.py +117 -0
  305. opentrons/protocol_engine/commands/calibration/calibrate_pipette.py +96 -0
  306. opentrons/protocol_engine/commands/calibration/move_to_maintenance_position.py +156 -0
  307. opentrons/protocol_engine/commands/command.py +308 -0
  308. opentrons/protocol_engine/commands/command_unions.py +974 -0
  309. opentrons/protocol_engine/commands/comment.py +57 -0
  310. opentrons/protocol_engine/commands/configure_for_volume.py +108 -0
  311. opentrons/protocol_engine/commands/configure_nozzle_layout.py +115 -0
  312. opentrons/protocol_engine/commands/custom.py +67 -0
  313. opentrons/protocol_engine/commands/dispense.py +194 -0
  314. opentrons/protocol_engine/commands/dispense_in_place.py +179 -0
  315. opentrons/protocol_engine/commands/dispense_while_tracking.py +204 -0
  316. opentrons/protocol_engine/commands/drop_tip.py +232 -0
  317. opentrons/protocol_engine/commands/drop_tip_in_place.py +205 -0
  318. opentrons/protocol_engine/commands/flex_stacker/__init__.py +64 -0
  319. opentrons/protocol_engine/commands/flex_stacker/common.py +900 -0
  320. opentrons/protocol_engine/commands/flex_stacker/empty.py +293 -0
  321. opentrons/protocol_engine/commands/flex_stacker/fill.py +281 -0
  322. opentrons/protocol_engine/commands/flex_stacker/retrieve.py +339 -0
  323. opentrons/protocol_engine/commands/flex_stacker/set_stored_labware.py +328 -0
  324. opentrons/protocol_engine/commands/flex_stacker/store.py +326 -0
  325. opentrons/protocol_engine/commands/generate_command_schema.py +61 -0
  326. opentrons/protocol_engine/commands/get_next_tip.py +134 -0
  327. opentrons/protocol_engine/commands/get_tip_presence.py +87 -0
  328. opentrons/protocol_engine/commands/hash_command_params.py +38 -0
  329. opentrons/protocol_engine/commands/heater_shaker/__init__.py +102 -0
  330. opentrons/protocol_engine/commands/heater_shaker/close_labware_latch.py +83 -0
  331. opentrons/protocol_engine/commands/heater_shaker/deactivate_heater.py +82 -0
  332. opentrons/protocol_engine/commands/heater_shaker/deactivate_shaker.py +84 -0
  333. opentrons/protocol_engine/commands/heater_shaker/open_labware_latch.py +110 -0
  334. opentrons/protocol_engine/commands/heater_shaker/set_and_wait_for_shake_speed.py +125 -0
  335. opentrons/protocol_engine/commands/heater_shaker/set_target_temperature.py +90 -0
  336. opentrons/protocol_engine/commands/heater_shaker/wait_for_temperature.py +102 -0
  337. opentrons/protocol_engine/commands/home.py +100 -0
  338. opentrons/protocol_engine/commands/identify_module.py +86 -0
  339. opentrons/protocol_engine/commands/labware_handling_common.py +29 -0
  340. opentrons/protocol_engine/commands/liquid_probe.py +464 -0
  341. opentrons/protocol_engine/commands/load_labware.py +210 -0
  342. opentrons/protocol_engine/commands/load_lid.py +154 -0
  343. opentrons/protocol_engine/commands/load_lid_stack.py +272 -0
  344. opentrons/protocol_engine/commands/load_liquid.py +95 -0
  345. opentrons/protocol_engine/commands/load_liquid_class.py +144 -0
  346. opentrons/protocol_engine/commands/load_module.py +223 -0
  347. opentrons/protocol_engine/commands/load_pipette.py +167 -0
  348. opentrons/protocol_engine/commands/magnetic_module/__init__.py +32 -0
  349. opentrons/protocol_engine/commands/magnetic_module/disengage.py +97 -0
  350. opentrons/protocol_engine/commands/magnetic_module/engage.py +119 -0
  351. opentrons/protocol_engine/commands/move_labware.py +546 -0
  352. opentrons/protocol_engine/commands/move_relative.py +102 -0
  353. opentrons/protocol_engine/commands/move_to_addressable_area.py +176 -0
  354. opentrons/protocol_engine/commands/move_to_addressable_area_for_drop_tip.py +198 -0
  355. opentrons/protocol_engine/commands/move_to_coordinates.py +107 -0
  356. opentrons/protocol_engine/commands/move_to_well.py +119 -0
  357. opentrons/protocol_engine/commands/movement_common.py +338 -0
  358. opentrons/protocol_engine/commands/pick_up_tip.py +241 -0
  359. opentrons/protocol_engine/commands/pipetting_common.py +443 -0
  360. opentrons/protocol_engine/commands/prepare_to_aspirate.py +121 -0
  361. opentrons/protocol_engine/commands/pressure_dispense.py +155 -0
  362. opentrons/protocol_engine/commands/reload_labware.py +90 -0
  363. opentrons/protocol_engine/commands/retract_axis.py +75 -0
  364. opentrons/protocol_engine/commands/robot/__init__.py +70 -0
  365. opentrons/protocol_engine/commands/robot/close_gripper_jaw.py +96 -0
  366. opentrons/protocol_engine/commands/robot/common.py +18 -0
  367. opentrons/protocol_engine/commands/robot/move_axes_relative.py +101 -0
  368. opentrons/protocol_engine/commands/robot/move_axes_to.py +100 -0
  369. opentrons/protocol_engine/commands/robot/move_to.py +94 -0
  370. opentrons/protocol_engine/commands/robot/open_gripper_jaw.py +86 -0
  371. opentrons/protocol_engine/commands/save_position.py +109 -0
  372. opentrons/protocol_engine/commands/seal_pipette_to_tip.py +353 -0
  373. opentrons/protocol_engine/commands/set_rail_lights.py +67 -0
  374. opentrons/protocol_engine/commands/set_status_bar.py +89 -0
  375. opentrons/protocol_engine/commands/temperature_module/__init__.py +46 -0
  376. opentrons/protocol_engine/commands/temperature_module/deactivate.py +86 -0
  377. opentrons/protocol_engine/commands/temperature_module/set_target_temperature.py +97 -0
  378. opentrons/protocol_engine/commands/temperature_module/wait_for_temperature.py +104 -0
  379. opentrons/protocol_engine/commands/thermocycler/__init__.py +152 -0
  380. opentrons/protocol_engine/commands/thermocycler/close_lid.py +87 -0
  381. opentrons/protocol_engine/commands/thermocycler/deactivate_block.py +80 -0
  382. opentrons/protocol_engine/commands/thermocycler/deactivate_lid.py +80 -0
  383. opentrons/protocol_engine/commands/thermocycler/open_lid.py +87 -0
  384. opentrons/protocol_engine/commands/thermocycler/run_extended_profile.py +171 -0
  385. opentrons/protocol_engine/commands/thermocycler/run_profile.py +124 -0
  386. opentrons/protocol_engine/commands/thermocycler/set_target_block_temperature.py +140 -0
  387. opentrons/protocol_engine/commands/thermocycler/set_target_lid_temperature.py +100 -0
  388. opentrons/protocol_engine/commands/thermocycler/wait_for_block_temperature.py +93 -0
  389. opentrons/protocol_engine/commands/thermocycler/wait_for_lid_temperature.py +89 -0
  390. opentrons/protocol_engine/commands/touch_tip.py +189 -0
  391. opentrons/protocol_engine/commands/unsafe/__init__.py +161 -0
  392. opentrons/protocol_engine/commands/unsafe/unsafe_blow_out_in_place.py +100 -0
  393. opentrons/protocol_engine/commands/unsafe/unsafe_drop_tip_in_place.py +121 -0
  394. opentrons/protocol_engine/commands/unsafe/unsafe_engage_axes.py +82 -0
  395. opentrons/protocol_engine/commands/unsafe/unsafe_place_labware.py +208 -0
  396. opentrons/protocol_engine/commands/unsafe/unsafe_stacker_close_latch.py +94 -0
  397. opentrons/protocol_engine/commands/unsafe/unsafe_stacker_manual_retrieve.py +295 -0
  398. opentrons/protocol_engine/commands/unsafe/unsafe_stacker_open_latch.py +91 -0
  399. opentrons/protocol_engine/commands/unsafe/unsafe_stacker_prepare_shuttle.py +136 -0
  400. opentrons/protocol_engine/commands/unsafe/unsafe_ungrip_labware.py +77 -0
  401. opentrons/protocol_engine/commands/unsafe/update_position_estimators.py +90 -0
  402. opentrons/protocol_engine/commands/unseal_pipette_from_tip.py +153 -0
  403. opentrons/protocol_engine/commands/verify_tip_presence.py +100 -0
  404. opentrons/protocol_engine/commands/wait_for_duration.py +76 -0
  405. opentrons/protocol_engine/commands/wait_for_resume.py +75 -0
  406. opentrons/protocol_engine/create_protocol_engine.py +193 -0
  407. opentrons/protocol_engine/engine_support.py +28 -0
  408. opentrons/protocol_engine/error_recovery_policy.py +81 -0
  409. opentrons/protocol_engine/errors/__init__.py +191 -0
  410. opentrons/protocol_engine/errors/error_occurrence.py +182 -0
  411. opentrons/protocol_engine/errors/exceptions.py +1308 -0
  412. opentrons/protocol_engine/execution/__init__.py +50 -0
  413. opentrons/protocol_engine/execution/command_executor.py +216 -0
  414. opentrons/protocol_engine/execution/create_queue_worker.py +102 -0
  415. opentrons/protocol_engine/execution/door_watcher.py +119 -0
  416. opentrons/protocol_engine/execution/equipment.py +819 -0
  417. opentrons/protocol_engine/execution/error_recovery_hardware_state_synchronizer.py +101 -0
  418. opentrons/protocol_engine/execution/gantry_mover.py +686 -0
  419. opentrons/protocol_engine/execution/hardware_stopper.py +147 -0
  420. opentrons/protocol_engine/execution/heater_shaker_movement_flagger.py +207 -0
  421. opentrons/protocol_engine/execution/labware_movement.py +297 -0
  422. opentrons/protocol_engine/execution/movement.py +349 -0
  423. opentrons/protocol_engine/execution/pipetting.py +607 -0
  424. opentrons/protocol_engine/execution/queue_worker.py +86 -0
  425. opentrons/protocol_engine/execution/rail_lights.py +25 -0
  426. opentrons/protocol_engine/execution/run_control.py +33 -0
  427. opentrons/protocol_engine/execution/status_bar.py +34 -0
  428. opentrons/protocol_engine/execution/thermocycler_movement_flagger.py +188 -0
  429. opentrons/protocol_engine/execution/thermocycler_plate_lifter.py +81 -0
  430. opentrons/protocol_engine/execution/tip_handler.py +550 -0
  431. opentrons/protocol_engine/labware_offset_standardization.py +194 -0
  432. opentrons/protocol_engine/notes/__init__.py +17 -0
  433. opentrons/protocol_engine/notes/notes.py +59 -0
  434. opentrons/protocol_engine/plugins.py +104 -0
  435. opentrons/protocol_engine/protocol_engine.py +683 -0
  436. opentrons/protocol_engine/resources/__init__.py +26 -0
  437. opentrons/protocol_engine/resources/deck_configuration_provider.py +232 -0
  438. opentrons/protocol_engine/resources/deck_data_provider.py +94 -0
  439. opentrons/protocol_engine/resources/file_provider.py +161 -0
  440. opentrons/protocol_engine/resources/fixture_validation.py +58 -0
  441. opentrons/protocol_engine/resources/labware_data_provider.py +106 -0
  442. opentrons/protocol_engine/resources/labware_validation.py +73 -0
  443. opentrons/protocol_engine/resources/model_utils.py +32 -0
  444. opentrons/protocol_engine/resources/module_data_provider.py +44 -0
  445. opentrons/protocol_engine/resources/ot3_validation.py +21 -0
  446. opentrons/protocol_engine/resources/pipette_data_provider.py +379 -0
  447. opentrons/protocol_engine/slot_standardization.py +128 -0
  448. opentrons/protocol_engine/state/__init__.py +1 -0
  449. opentrons/protocol_engine/state/_abstract_store.py +27 -0
  450. opentrons/protocol_engine/state/_axis_aligned_bounding_box.py +50 -0
  451. opentrons/protocol_engine/state/_labware_origin_math.py +636 -0
  452. opentrons/protocol_engine/state/_move_types.py +83 -0
  453. opentrons/protocol_engine/state/_well_math.py +193 -0
  454. opentrons/protocol_engine/state/addressable_areas.py +699 -0
  455. opentrons/protocol_engine/state/command_history.py +309 -0
  456. opentrons/protocol_engine/state/commands.py +1158 -0
  457. opentrons/protocol_engine/state/config.py +39 -0
  458. opentrons/protocol_engine/state/files.py +57 -0
  459. opentrons/protocol_engine/state/fluid_stack.py +138 -0
  460. opentrons/protocol_engine/state/geometry.py +2359 -0
  461. opentrons/protocol_engine/state/inner_well_math_utils.py +548 -0
  462. opentrons/protocol_engine/state/labware.py +1459 -0
  463. opentrons/protocol_engine/state/liquid_classes.py +82 -0
  464. opentrons/protocol_engine/state/liquids.py +73 -0
  465. opentrons/protocol_engine/state/module_substates/__init__.py +45 -0
  466. opentrons/protocol_engine/state/module_substates/absorbance_reader_substate.py +35 -0
  467. opentrons/protocol_engine/state/module_substates/flex_stacker_substate.py +112 -0
  468. opentrons/protocol_engine/state/module_substates/heater_shaker_module_substate.py +115 -0
  469. opentrons/protocol_engine/state/module_substates/magnetic_block_substate.py +17 -0
  470. opentrons/protocol_engine/state/module_substates/magnetic_module_substate.py +65 -0
  471. opentrons/protocol_engine/state/module_substates/temperature_module_substate.py +67 -0
  472. opentrons/protocol_engine/state/module_substates/thermocycler_module_substate.py +163 -0
  473. opentrons/protocol_engine/state/modules.py +1500 -0
  474. opentrons/protocol_engine/state/motion.py +373 -0
  475. opentrons/protocol_engine/state/pipettes.py +905 -0
  476. opentrons/protocol_engine/state/state.py +421 -0
  477. opentrons/protocol_engine/state/state_summary.py +36 -0
  478. opentrons/protocol_engine/state/tips.py +420 -0
  479. opentrons/protocol_engine/state/update_types.py +904 -0
  480. opentrons/protocol_engine/state/wells.py +290 -0
  481. opentrons/protocol_engine/types/__init__.py +308 -0
  482. opentrons/protocol_engine/types/automatic_tip_selection.py +39 -0
  483. opentrons/protocol_engine/types/command_annotations.py +53 -0
  484. opentrons/protocol_engine/types/deck_configuration.py +81 -0
  485. opentrons/protocol_engine/types/execution.py +96 -0
  486. opentrons/protocol_engine/types/hardware_passthrough.py +25 -0
  487. opentrons/protocol_engine/types/instrument.py +47 -0
  488. opentrons/protocol_engine/types/instrument_sensors.py +47 -0
  489. opentrons/protocol_engine/types/labware.py +131 -0
  490. opentrons/protocol_engine/types/labware_movement.py +22 -0
  491. opentrons/protocol_engine/types/labware_offset_location.py +111 -0
  492. opentrons/protocol_engine/types/labware_offset_vector.py +16 -0
  493. opentrons/protocol_engine/types/liquid.py +40 -0
  494. opentrons/protocol_engine/types/liquid_class.py +59 -0
  495. opentrons/protocol_engine/types/liquid_handling.py +13 -0
  496. opentrons/protocol_engine/types/liquid_level_detection.py +191 -0
  497. opentrons/protocol_engine/types/location.py +194 -0
  498. opentrons/protocol_engine/types/module.py +303 -0
  499. opentrons/protocol_engine/types/partial_tip_configuration.py +76 -0
  500. opentrons/protocol_engine/types/run_time_parameters.py +133 -0
  501. opentrons/protocol_engine/types/tip.py +18 -0
  502. opentrons/protocol_engine/types/util.py +21 -0
  503. opentrons/protocol_engine/types/well_position.py +124 -0
  504. opentrons/protocol_reader/__init__.py +37 -0
  505. opentrons/protocol_reader/extract_labware_definitions.py +66 -0
  506. opentrons/protocol_reader/file_format_validator.py +152 -0
  507. opentrons/protocol_reader/file_hasher.py +27 -0
  508. opentrons/protocol_reader/file_identifier.py +284 -0
  509. opentrons/protocol_reader/file_reader_writer.py +90 -0
  510. opentrons/protocol_reader/input_file.py +16 -0
  511. opentrons/protocol_reader/protocol_files_invalid_error.py +6 -0
  512. opentrons/protocol_reader/protocol_reader.py +188 -0
  513. opentrons/protocol_reader/protocol_source.py +124 -0
  514. opentrons/protocol_reader/role_analyzer.py +86 -0
  515. opentrons/protocol_runner/__init__.py +26 -0
  516. opentrons/protocol_runner/create_simulating_orchestrator.py +118 -0
  517. opentrons/protocol_runner/json_file_reader.py +55 -0
  518. opentrons/protocol_runner/json_translator.py +314 -0
  519. opentrons/protocol_runner/legacy_command_mapper.py +848 -0
  520. opentrons/protocol_runner/legacy_context_plugin.py +116 -0
  521. opentrons/protocol_runner/protocol_runner.py +530 -0
  522. opentrons/protocol_runner/python_protocol_wrappers.py +179 -0
  523. opentrons/protocol_runner/run_orchestrator.py +496 -0
  524. opentrons/protocol_runner/task_queue.py +95 -0
  525. opentrons/protocols/__init__.py +6 -0
  526. opentrons/protocols/advanced_control/__init__.py +0 -0
  527. opentrons/protocols/advanced_control/common.py +38 -0
  528. opentrons/protocols/advanced_control/mix.py +60 -0
  529. opentrons/protocols/advanced_control/transfers/__init__.py +0 -0
  530. opentrons/protocols/advanced_control/transfers/common.py +180 -0
  531. opentrons/protocols/advanced_control/transfers/transfer.py +972 -0
  532. opentrons/protocols/advanced_control/transfers/transfer_liquid_utils.py +231 -0
  533. opentrons/protocols/api_support/__init__.py +0 -0
  534. opentrons/protocols/api_support/constants.py +8 -0
  535. opentrons/protocols/api_support/deck_type.py +110 -0
  536. opentrons/protocols/api_support/definitions.py +18 -0
  537. opentrons/protocols/api_support/instrument.py +151 -0
  538. opentrons/protocols/api_support/labware_like.py +233 -0
  539. opentrons/protocols/api_support/tip_tracker.py +175 -0
  540. opentrons/protocols/api_support/types.py +32 -0
  541. opentrons/protocols/api_support/util.py +403 -0
  542. opentrons/protocols/bundle.py +89 -0
  543. opentrons/protocols/duration/__init__.py +4 -0
  544. opentrons/protocols/duration/errors.py +5 -0
  545. opentrons/protocols/duration/estimator.py +628 -0
  546. opentrons/protocols/execution/__init__.py +0 -0
  547. opentrons/protocols/execution/dev_types.py +181 -0
  548. opentrons/protocols/execution/errors.py +40 -0
  549. opentrons/protocols/execution/execute.py +84 -0
  550. opentrons/protocols/execution/execute_json_v3.py +275 -0
  551. opentrons/protocols/execution/execute_json_v4.py +359 -0
  552. opentrons/protocols/execution/execute_json_v5.py +28 -0
  553. opentrons/protocols/execution/execute_python.py +169 -0
  554. opentrons/protocols/execution/json_dispatchers.py +87 -0
  555. opentrons/protocols/execution/types.py +7 -0
  556. opentrons/protocols/geometry/__init__.py +0 -0
  557. opentrons/protocols/geometry/planning.py +297 -0
  558. opentrons/protocols/labware.py +312 -0
  559. opentrons/protocols/models/__init__.py +0 -0
  560. opentrons/protocols/models/json_protocol.py +679 -0
  561. opentrons/protocols/parameters/__init__.py +0 -0
  562. opentrons/protocols/parameters/csv_parameter_definition.py +77 -0
  563. opentrons/protocols/parameters/csv_parameter_interface.py +96 -0
  564. opentrons/protocols/parameters/exceptions.py +34 -0
  565. opentrons/protocols/parameters/parameter_definition.py +272 -0
  566. opentrons/protocols/parameters/types.py +17 -0
  567. opentrons/protocols/parameters/validation.py +267 -0
  568. opentrons/protocols/parse.py +671 -0
  569. opentrons/protocols/types.py +159 -0
  570. opentrons/py.typed +0 -0
  571. opentrons/resources/scripts/lpc21isp +0 -0
  572. opentrons/resources/smoothie-edge-8414642.hex +23010 -0
  573. opentrons/simulate.py +1065 -0
  574. opentrons/system/__init__.py +6 -0
  575. opentrons/system/camera.py +51 -0
  576. opentrons/system/log_control.py +59 -0
  577. opentrons/system/nmcli.py +856 -0
  578. opentrons/system/resin.py +24 -0
  579. opentrons/system/smoothie_update.py +15 -0
  580. opentrons/system/wifi.py +204 -0
  581. opentrons/tools/__init__.py +0 -0
  582. opentrons/tools/args_handler.py +22 -0
  583. opentrons/tools/write_pipette_memory.py +157 -0
  584. opentrons/types.py +618 -0
  585. opentrons/util/__init__.py +1 -0
  586. opentrons/util/async_helpers.py +166 -0
  587. opentrons/util/broker.py +84 -0
  588. opentrons/util/change_notifier.py +47 -0
  589. opentrons/util/entrypoint_util.py +278 -0
  590. opentrons/util/get_union_elements.py +26 -0
  591. opentrons/util/helpers.py +6 -0
  592. opentrons/util/linal.py +178 -0
  593. opentrons/util/logging_config.py +265 -0
  594. opentrons/util/logging_queue_handler.py +61 -0
  595. opentrons/util/performance_helpers.py +157 -0
  596. opentrons-8.6.0a1.dist-info/METADATA +37 -0
  597. opentrons-8.6.0a1.dist-info/RECORD +600 -0
  598. opentrons-8.6.0a1.dist-info/WHEEL +4 -0
  599. opentrons-8.6.0a1.dist-info/entry_points.txt +3 -0
  600. opentrons-8.6.0a1.dist-info/licenses/LICENSE +202 -0
@@ -0,0 +1,848 @@
1
+ """Translate events from a legacy ``ProtocolContext`` into Protocol Engine commands."""
2
+
3
+ from collections import defaultdict
4
+ from datetime import datetime
5
+ from typing import Dict, List, Optional, Tuple, Union
6
+
7
+ from opentrons.hardware_control.modules.types import (
8
+ ModuleModel as HardwareModuleModel,
9
+ TemperatureModuleModel,
10
+ MagneticModuleModel,
11
+ ThermocyclerModuleModel,
12
+ HeaterShakerModuleModel,
13
+ )
14
+ from opentrons_shared_data.pipette.types import PipetteNameType
15
+ from opentrons.types import MountType, DeckSlotName, Location
16
+ from opentrons.legacy_commands import types as legacy_command_types
17
+ from opentrons.protocol_api import InstrumentContext
18
+ from opentrons.protocol_api.core.legacy.deck import FIXED_TRASH_ID
19
+ from opentrons.protocol_api.core.legacy.load_info import (
20
+ LoadInfo as LegacyLoadInfo,
21
+ LabwareLoadInfo as LegacyLabwareLoadInfo,
22
+ InstrumentLoadInfo as LegacyInstrumentLoadInfo,
23
+ ModuleLoadInfo as LegacyModuleLoadInfo,
24
+ )
25
+ from opentrons.protocol_engine import (
26
+ ProtocolEngineError,
27
+ actions as pe_actions,
28
+ commands as pe_commands,
29
+ types as pe_types,
30
+ )
31
+ from opentrons.protocol_engine.error_recovery_policy import ErrorRecoveryType
32
+ from opentrons.protocol_engine.resources import (
33
+ ModelUtils,
34
+ ModuleDataProvider,
35
+ pipette_data_provider,
36
+ )
37
+ from opentrons.protocol_engine.state.update_types import (
38
+ StateUpdate,
39
+ )
40
+
41
+ from opentrons_shared_data.labware.labware_definition import (
42
+ labware_definition_type_adapter,
43
+ )
44
+ from opentrons_shared_data.errors import ErrorCodes, EnumeratedError, PythonException
45
+
46
+
47
+ class LegacyCommandParams(pe_commands.CustomParams):
48
+ """Custom command data payload for mapped legacy commands."""
49
+
50
+ legacyCommandType: str
51
+ legacyCommandText: str
52
+
53
+
54
+ class LegacyContextCommandError(ProtocolEngineError):
55
+ """An error returned when a PAPIv2 ProtocolContext command fails."""
56
+
57
+ def __init__(self, wrapping_exc: BaseException) -> None:
58
+ if isinstance(wrapping_exc, EnumeratedError):
59
+ super().__init__(
60
+ wrapping_exc.code,
61
+ wrapping_exc.message,
62
+ wrapping_exc.detail,
63
+ wrapping_exc.wrapping,
64
+ )
65
+ else:
66
+ super().__init__(
67
+ code=ErrorCodes.GENERAL_ERROR,
68
+ message=str(wrapping_exc),
69
+ wrapping=[PythonException(wrapping_exc)],
70
+ )
71
+
72
+
73
+ _HARDWARE_TO_PE_MODULE: Dict[HardwareModuleModel, pe_types.ModuleModel] = {
74
+ MagneticModuleModel.MAGNETIC_V1: pe_types.ModuleModel.MAGNETIC_MODULE_V1,
75
+ MagneticModuleModel.MAGNETIC_V2: pe_types.ModuleModel.MAGNETIC_MODULE_V2,
76
+ TemperatureModuleModel.TEMPERATURE_V1: pe_types.ModuleModel.TEMPERATURE_MODULE_V1,
77
+ TemperatureModuleModel.TEMPERATURE_V2: pe_types.ModuleModel.TEMPERATURE_MODULE_V2,
78
+ ThermocyclerModuleModel.THERMOCYCLER_V1: pe_types.ModuleModel.THERMOCYCLER_MODULE_V1,
79
+ ThermocyclerModuleModel.THERMOCYCLER_V2: pe_types.ModuleModel.THERMOCYCLER_MODULE_V2,
80
+ HeaterShakerModuleModel.HEATER_SHAKER_V1: pe_types.ModuleModel.HEATER_SHAKER_MODULE_V1,
81
+ }
82
+
83
+ _HIGHER_ORDER_COMMAND_TYPES = {
84
+ legacy_command_types.MIX,
85
+ legacy_command_types.CONSOLIDATE,
86
+ legacy_command_types.DISTRIBUTE,
87
+ legacy_command_types.TRANSFER,
88
+ legacy_command_types.RETURN_TIP,
89
+ legacy_command_types.AIR_GAP,
90
+ }
91
+
92
+
93
+ class LegacyCommandMapper:
94
+ """Map broker commands to protocol engine commands.
95
+
96
+ Each protocol should use its own instance of this class.
97
+ """
98
+
99
+ def __init__(
100
+ self, module_data_provider: Optional[ModuleDataProvider] = None
101
+ ) -> None:
102
+ """Initialize the command mapper."""
103
+ # commands keyed by broker message ID
104
+ self._commands_by_broker_id: Dict[str, pe_commands.Command] = {}
105
+
106
+ # running count of each legacy command type, to construct IDs
107
+ self._command_count: Dict[str, int] = defaultdict(lambda: 0)
108
+
109
+ # equipment IDs by physical location
110
+ self._labware_id_by_slot: Dict[DeckSlotName, str] = {
111
+ DeckSlotName.FIXED_TRASH: FIXED_TRASH_ID
112
+ }
113
+ self._labware_id_by_module_id: Dict[str, str] = {}
114
+ self._pipette_id_by_mount: Dict[MountType, str] = {}
115
+ self._module_id_by_slot: Dict[DeckSlotName, str] = {}
116
+
117
+ # module definition state and provider depedency
118
+ self._module_definition_by_model: Dict[
119
+ pe_types.ModuleModel, pe_types.ModuleDefinition
120
+ ] = {}
121
+ self._module_data_provider = module_data_provider or ModuleDataProvider()
122
+
123
+ def map_command( # noqa: C901
124
+ self,
125
+ command: legacy_command_types.CommandMessage,
126
+ ) -> List[pe_actions.Action]:
127
+ """Map a legacy Broker command to a ProtocolEngine command.
128
+
129
+ A "before" message from the Broker
130
+ is mapped to a ``RUNNING`` ProtocolEngine command.
131
+
132
+ An "after" message from the Broker
133
+ is mapped to a ``SUCCEEDED`` ProtocolEngine command.
134
+ It has the same ID as the original ``RUNNING`` command,
135
+ so when you send it to the ProtocolEngine, it will update the original
136
+ command's status in-place.
137
+ """
138
+ command_type = command["name"]
139
+
140
+ if command_type in _HIGHER_ORDER_COMMAND_TYPES:
141
+ return []
142
+
143
+ command_error = command["error"]
144
+ stage = command["$"]
145
+ # TODO(mc, 2021-12-08): use message ID as command ID directly once
146
+ # https://github.com/Opentrons/opentrons/issues/8986 is resolved
147
+ broker_id = command["id"]
148
+ now = ModelUtils.get_timestamp()
149
+
150
+ results: List[pe_actions.Action] = []
151
+
152
+ if stage == "before":
153
+ count = self._command_count[command_type]
154
+ command_id = f"{command_type}-{count}"
155
+ command_create, running_command = self._build_initial_command(
156
+ command, command_id, now
157
+ )
158
+
159
+ self._command_count[command_type] = count + 1
160
+ self._commands_by_broker_id[broker_id] = running_command
161
+
162
+ results.append(
163
+ pe_actions.QueueCommandAction(
164
+ command_id=command_id,
165
+ created_at=running_command.createdAt,
166
+ request=command_create,
167
+ request_hash=None,
168
+ )
169
+ )
170
+ assert running_command.startedAt is not None
171
+ results.append(
172
+ pe_actions.RunCommandAction(
173
+ running_command.id, started_at=running_command.startedAt
174
+ )
175
+ )
176
+
177
+ elif stage == "after":
178
+ running_command = self._commands_by_broker_id[broker_id]
179
+ completed_command: pe_commands.Command
180
+ if command_error is None:
181
+ if isinstance(running_command, pe_commands.PickUpTip):
182
+ completed_command = running_command.model_copy(
183
+ update={
184
+ "result": pe_commands.PickUpTipResult.model_construct(
185
+ tipVolume=command["payload"]["location"].max_volume, # type: ignore[typeddict-item]
186
+ tipLength=command["payload"]["instrument"].hw_pipette[ # type: ignore[typeddict-item]
187
+ "tip_length"
188
+ ],
189
+ position=pe_types.DeckPoint(x=0, y=0, z=0),
190
+ ),
191
+ "status": pe_commands.CommandStatus.SUCCEEDED,
192
+ "completedAt": now,
193
+ "notes": [],
194
+ }
195
+ )
196
+ elif isinstance(running_command, pe_commands.DropTip):
197
+ completed_command = running_command.model_copy(
198
+ update={
199
+ "result": pe_commands.DropTipResult.model_construct(
200
+ position=pe_types.DeckPoint(x=0, y=0, z=0)
201
+ ),
202
+ "status": pe_commands.CommandStatus.SUCCEEDED,
203
+ "completedAt": now,
204
+ "notes": [],
205
+ }
206
+ )
207
+ elif isinstance(running_command, pe_commands.Aspirate):
208
+ completed_command = running_command.model_copy(
209
+ update={
210
+ # Don't .model_construct() result, because we want to validate
211
+ # volume.
212
+ "result": pe_commands.AspirateResult(
213
+ volume=running_command.params.volume,
214
+ position=pe_types.DeckPoint(x=0, y=0, z=0),
215
+ ),
216
+ "status": pe_commands.CommandStatus.SUCCEEDED,
217
+ "completedAt": now,
218
+ "notes": [],
219
+ }
220
+ )
221
+ elif isinstance(running_command, pe_commands.Dispense):
222
+ completed_command = running_command.model_copy(
223
+ update={
224
+ # Don't .model_construct() result, because we want to validate
225
+ # volume.
226
+ "result": pe_commands.DispenseResult(
227
+ volume=running_command.params.volume,
228
+ position=pe_types.DeckPoint(x=0, y=0, z=0),
229
+ ),
230
+ "status": pe_commands.CommandStatus.SUCCEEDED,
231
+ "completedAt": now,
232
+ "notes": [],
233
+ }
234
+ )
235
+ elif isinstance(running_command, pe_commands.BlowOut):
236
+ completed_command = running_command.model_copy(
237
+ update={
238
+ "result": pe_commands.BlowOutResult.model_construct(
239
+ position=pe_types.DeckPoint(x=0, y=0, z=0)
240
+ ),
241
+ "status": pe_commands.CommandStatus.SUCCEEDED,
242
+ "completedAt": now,
243
+ "notes": [],
244
+ }
245
+ )
246
+ elif isinstance(running_command, pe_commands.Comment):
247
+ completed_command = running_command.model_copy(
248
+ update={
249
+ "result": pe_commands.CommentResult.model_construct(),
250
+ "status": pe_commands.CommandStatus.SUCCEEDED,
251
+ "completedAt": now,
252
+ "notes": [],
253
+ }
254
+ )
255
+ elif isinstance(running_command, pe_commands.Custom):
256
+ completed_command = running_command.model_copy(
257
+ update={
258
+ "result": pe_commands.CustomResult.model_construct(),
259
+ "status": pe_commands.CommandStatus.SUCCEEDED,
260
+ "completedAt": now,
261
+ "notes": [],
262
+ }
263
+ )
264
+ else:
265
+ # TODO(mm, 2024-06-13): This looks potentially wrong.
266
+ # We're creating a `SUCCEEDED` command that does not have a `result`,
267
+ # which is not normally possible.
268
+ completed_command = running_command.model_copy(
269
+ update={
270
+ "status": pe_commands.CommandStatus.SUCCEEDED,
271
+ "completedAt": now,
272
+ "notes": [],
273
+ }
274
+ )
275
+ results.append(
276
+ pe_actions.SucceedCommandAction(
277
+ completed_command,
278
+ state_update=StateUpdate(),
279
+ )
280
+ )
281
+
282
+ if isinstance(completed_command, pe_commands.WaitForResume):
283
+ results.append(
284
+ pe_actions.PauseAction(source=pe_actions.PauseSource.PROTOCOL)
285
+ )
286
+
287
+ else:
288
+ results.append(
289
+ pe_actions.FailCommandAction(
290
+ command_id=running_command.id,
291
+ running_command=running_command,
292
+ error_id=ModelUtils.generate_id(),
293
+ failed_at=now,
294
+ error=LegacyContextCommandError(command_error),
295
+ notes=[],
296
+ # For legacy protocols, we don't attempt to support any kind
297
+ # of error recovery at the Protocol Engine level.
298
+ # These protocols only run on the OT-2, which doesn't have
299
+ # any recoverable errors, anyway.
300
+ type=ErrorRecoveryType.FAIL_RUN,
301
+ )
302
+ )
303
+
304
+ return results
305
+
306
+ def map_equipment_load(self, load_info: LegacyLoadInfo) -> List[pe_actions.Action]:
307
+ """Map a labware, instrument (pipette), or module load to a PE command."""
308
+ if isinstance(load_info, LegacyLabwareLoadInfo):
309
+ return self._map_labware_load(load_info)
310
+ elif isinstance(load_info, LegacyInstrumentLoadInfo):
311
+ return self._map_instrument_load(load_info)
312
+ elif isinstance(load_info, LegacyModuleLoadInfo):
313
+ return self._map_module_load(load_info)
314
+
315
+ def _build_initial_command(
316
+ self,
317
+ command: legacy_command_types.CommandMessage,
318
+ command_id: str,
319
+ now: datetime,
320
+ ) -> Tuple[pe_commands.CommandCreate, pe_commands.Command]:
321
+ if command["name"] == legacy_command_types.PICK_UP_TIP:
322
+ return self._build_pick_up_tip(
323
+ command=command, command_id=command_id, now=now
324
+ )
325
+ elif command["name"] == legacy_command_types.DROP_TIP:
326
+ return self._build_drop_tip(command=command, command_id=command_id, now=now)
327
+
328
+ elif (
329
+ command["name"] == legacy_command_types.ASPIRATE
330
+ or command["name"] == legacy_command_types.DISPENSE
331
+ ):
332
+ return self._build_liquid_handling(
333
+ command=command, command_id=command_id, now=now
334
+ )
335
+ elif command["name"] == legacy_command_types.BLOW_OUT:
336
+ return self._build_blow_out(command=command, command_id=command_id, now=now)
337
+ elif command["name"] == legacy_command_types.PAUSE:
338
+ wait_for_resume_running = pe_commands.WaitForResume.model_construct(
339
+ id=command_id,
340
+ key=command_id,
341
+ status=pe_commands.CommandStatus.RUNNING,
342
+ createdAt=now,
343
+ startedAt=now,
344
+ params=pe_commands.WaitForResumeParams.model_construct(
345
+ message=command["payload"]["userMessage"],
346
+ ),
347
+ )
348
+ wait_for_resume_create: pe_commands.CommandCreate = (
349
+ pe_commands.WaitForResumeCreate.model_construct(
350
+ key=wait_for_resume_running.key,
351
+ params=wait_for_resume_running.params,
352
+ )
353
+ )
354
+ return wait_for_resume_create, wait_for_resume_running
355
+ elif command["name"] == legacy_command_types.COMMENT:
356
+ comment_running = pe_commands.Comment.model_construct(
357
+ id=command_id,
358
+ key=command_id,
359
+ status=pe_commands.CommandStatus.RUNNING,
360
+ createdAt=now,
361
+ startedAt=now,
362
+ params=pe_commands.CommentParams.model_construct(
363
+ message=command["payload"]["text"],
364
+ ),
365
+ )
366
+ comment_create = pe_commands.CommentCreate.model_construct(
367
+ key=comment_running.key, params=comment_running.params
368
+ )
369
+ return comment_create, comment_running
370
+ else:
371
+ custom_running = pe_commands.Custom.model_construct(
372
+ id=command_id,
373
+ key=command_id,
374
+ status=pe_commands.CommandStatus.RUNNING,
375
+ createdAt=now,
376
+ startedAt=now,
377
+ params=LegacyCommandParams.model_construct(
378
+ legacyCommandType=command["name"],
379
+ legacyCommandText=command["payload"]["text"],
380
+ ),
381
+ )
382
+ custom_create = pe_commands.CustomCreate.model_construct(
383
+ key=custom_running.key,
384
+ params=custom_running.params,
385
+ )
386
+ return custom_create, custom_running
387
+
388
+ def _build_drop_tip(
389
+ self,
390
+ command: legacy_command_types.DropTipMessage,
391
+ command_id: str,
392
+ now: datetime,
393
+ ) -> Tuple[pe_commands.CommandCreate, pe_commands.Command]:
394
+ pipette: InstrumentContext = command["payload"]["instrument"]
395
+ well = command["payload"]["location"]
396
+ mount = MountType(pipette.mount)
397
+ # the following type checking suppression assumes the tiprack is not loaded on top of a module
398
+ slot = DeckSlotName.from_primitive(well.parent.parent) # type: ignore[arg-type]
399
+ well_name = well.well_name
400
+ labware_id = self._labware_id_by_slot[slot]
401
+ pipette_id = self._pipette_id_by_mount[mount]
402
+
403
+ running = pe_commands.DropTip.model_construct(
404
+ id=command_id,
405
+ key=command_id,
406
+ status=pe_commands.CommandStatus.RUNNING,
407
+ createdAt=now,
408
+ startedAt=now,
409
+ params=pe_commands.DropTipParams.model_construct(
410
+ pipetteId=pipette_id,
411
+ labwareId=labware_id,
412
+ wellName=well_name,
413
+ ),
414
+ )
415
+ create = pe_commands.DropTipCreate.model_construct(
416
+ key=running.key,
417
+ params=running.params,
418
+ )
419
+ return create, running
420
+
421
+ def _build_pick_up_tip(
422
+ self,
423
+ command: legacy_command_types.PickUpTipMessage,
424
+ command_id: str,
425
+ now: datetime,
426
+ ) -> Tuple[pe_commands.CommandCreate, pe_commands.Command]:
427
+ pipette: InstrumentContext = command["payload"]["instrument"]
428
+ location = command["payload"]["location"]
429
+ well = location
430
+ mount = MountType(pipette.mount)
431
+ # the following type checking suppression assumes the tiprack is not loaded on top of a module
432
+ slot = DeckSlotName.from_primitive(well.parent.parent) # type: ignore[arg-type]
433
+ well_name = well.well_name
434
+ labware_id = self._labware_id_by_slot[slot]
435
+ pipette_id = self._pipette_id_by_mount[mount]
436
+
437
+ running = pe_commands.PickUpTip.model_construct(
438
+ id=command_id,
439
+ key=command_id,
440
+ status=pe_commands.CommandStatus.RUNNING,
441
+ createdAt=now,
442
+ startedAt=now,
443
+ params=pe_commands.PickUpTipParams.model_construct(
444
+ pipetteId=pipette_id,
445
+ labwareId=labware_id,
446
+ wellName=well_name,
447
+ ),
448
+ )
449
+ create = pe_commands.PickUpTipCreate.model_construct(
450
+ key=running.key, params=running.params
451
+ )
452
+ return create, running
453
+
454
+ def _build_liquid_handling(
455
+ self,
456
+ command: Union[
457
+ legacy_command_types.AspirateMessage, legacy_command_types.DispenseMessage
458
+ ],
459
+ command_id: str,
460
+ now: datetime,
461
+ ) -> Tuple[pe_commands.CommandCreate, pe_commands.Command]:
462
+ pipette: InstrumentContext = command["payload"]["instrument"]
463
+ location = command["payload"]["location"]
464
+ volume = command["payload"]["volume"]
465
+ # TODO:(jr, 15.08.2022): aspirate and dispense commands with no specified labware
466
+ # get filtered into custom. Refactor this in followup legacy command mapping
467
+ if location.labware.is_well:
468
+ well = location.labware.as_well()
469
+ slot = DeckSlotName(location.labware.first_parent())
470
+ parent_module_id = self._module_id_by_slot.get(slot)
471
+ labware_id = (
472
+ self._labware_id_by_module_id[parent_module_id]
473
+ if parent_module_id is not None
474
+ else self._labware_id_by_slot[slot]
475
+ )
476
+ mount = MountType(pipette.mount)
477
+ well_name = well.well_name
478
+ pipette_id = self._pipette_id_by_mount[mount]
479
+
480
+ if volume == 0:
481
+ # In edge cases, it's possible for a Python protocol to do dispense()
482
+ # or aspirate() with a volume of 0, which behaves roughly like
483
+ # move_to(). Protocol Engine aspirate and dispense commands must have
484
+ # volume > 0, so we can't map into those.
485
+ #
486
+ # TODO(mm, 2024-03-22): I don't think this has been true since
487
+ # https://github.com/Opentrons/opentrons/pull/14211. Can we just use
488
+ # aspirate and dispense commands now?
489
+ move_to_well_running = pe_commands.MoveToWell.model_construct(
490
+ id=command_id,
491
+ key=command_id,
492
+ status=pe_commands.CommandStatus.RUNNING,
493
+ createdAt=now,
494
+ startedAt=now,
495
+ params=pe_commands.MoveToWellParams.model_construct(
496
+ pipetteId=pipette_id,
497
+ labwareId=labware_id,
498
+ wellName=well_name,
499
+ ),
500
+ )
501
+ move_to_well_create = pe_commands.MoveToWellCreate.model_construct(
502
+ key=move_to_well_running.key, params=move_to_well_running.params
503
+ )
504
+ return move_to_well_create, move_to_well_running
505
+ elif command["name"] == legacy_command_types.ASPIRATE:
506
+ flow_rate = command["payload"]["rate"] * pipette.flow_rate.aspirate
507
+ aspirate_running = pe_commands.Aspirate.model_construct(
508
+ id=command_id,
509
+ key=command_id,
510
+ status=pe_commands.CommandStatus.RUNNING,
511
+ createdAt=now,
512
+ startedAt=now,
513
+ # Don't .model_construct() params, because we want to validate
514
+ # volume and flowRate.
515
+ params=pe_commands.AspirateParams(
516
+ pipetteId=pipette_id,
517
+ labwareId=labware_id,
518
+ wellName=well_name,
519
+ volume=volume,
520
+ flowRate=flow_rate,
521
+ ),
522
+ )
523
+ aspirate_create = pe_commands.AspirateCreate.model_construct(
524
+ key=aspirate_running.key, params=aspirate_running.params
525
+ )
526
+ return aspirate_create, aspirate_running
527
+ else:
528
+ flow_rate = command["payload"]["rate"] * pipette.flow_rate.dispense
529
+ dispense_running = pe_commands.Dispense.model_construct(
530
+ id=command_id,
531
+ key=command_id,
532
+ status=pe_commands.CommandStatus.RUNNING,
533
+ createdAt=now,
534
+ startedAt=now,
535
+ # Don't .model_construct params, because we want to validate
536
+ # volume and flowRate.
537
+ params=pe_commands.DispenseParams(
538
+ pipetteId=pipette_id,
539
+ labwareId=labware_id,
540
+ wellName=well_name,
541
+ volume=volume,
542
+ flowRate=flow_rate,
543
+ ),
544
+ )
545
+ dispense_create = pe_commands.DispenseCreate.model_construct(
546
+ key=dispense_running.key, params=dispense_running.params
547
+ )
548
+ return dispense_create, dispense_running
549
+
550
+ else:
551
+ running = pe_commands.Custom.model_construct(
552
+ id=command_id,
553
+ key=command_id,
554
+ status=pe_commands.CommandStatus.RUNNING,
555
+ createdAt=now,
556
+ startedAt=now,
557
+ params=LegacyCommandParams.model_construct(
558
+ legacyCommandType=command["name"],
559
+ legacyCommandText=command["payload"]["text"],
560
+ ),
561
+ )
562
+ create = pe_commands.CustomCreate.model_construct(
563
+ key=running.key, params=running.params
564
+ )
565
+ return create, running
566
+
567
+ def _build_blow_out(
568
+ self,
569
+ command: legacy_command_types.BlowOutMessage,
570
+ command_id: str,
571
+ now: datetime,
572
+ ) -> Tuple[pe_commands.CommandCreate, pe_commands.Command]:
573
+ pipette: InstrumentContext = command["payload"]["instrument"]
574
+ location = command["payload"]["location"]
575
+ flow_rate = pipette.flow_rate.blow_out
576
+ # TODO:(jr, 15.08.2022): blow_out commands with no specified labware get filtered
577
+ # into custom. Remove location.labware.is_empty is False when refactor is complete
578
+ if isinstance(location, Location) and location.labware.is_well:
579
+ well = location.labware.as_well()
580
+ slot = DeckSlotName(location.labware.first_parent())
581
+ parent_module_id = self._module_id_by_slot.get(slot)
582
+ labware_id = (
583
+ self._labware_id_by_module_id[parent_module_id]
584
+ if parent_module_id is not None
585
+ else self._labware_id_by_slot[slot]
586
+ )
587
+ mount = MountType(pipette.mount)
588
+ well_name = well.well_name
589
+ pipette_id = self._pipette_id_by_mount[mount]
590
+
591
+ blow_out_running = pe_commands.BlowOut.model_construct(
592
+ id=command_id,
593
+ key=command_id,
594
+ status=pe_commands.CommandStatus.RUNNING,
595
+ createdAt=now,
596
+ startedAt=now,
597
+ # Don't .model_construct() params, because we want to validate flowRate.
598
+ params=pe_commands.BlowOutParams(
599
+ pipetteId=pipette_id,
600
+ labwareId=labware_id,
601
+ wellName=well_name,
602
+ flowRate=flow_rate,
603
+ ),
604
+ )
605
+ blow_out_create = pe_commands.BlowOutCreate.model_construct(
606
+ key=blow_out_running.key, params=blow_out_running.params
607
+ )
608
+ return blow_out_create, blow_out_running
609
+
610
+ # TODO:(jr, 15.08.2022): blow_out commands with no specified labware get filtered
611
+ # into custom. Refactor this in followup legacy command mapping
612
+ else:
613
+ custom_running = pe_commands.Custom.model_construct(
614
+ id=command_id,
615
+ key=command_id,
616
+ status=pe_commands.CommandStatus.RUNNING,
617
+ createdAt=now,
618
+ startedAt=now,
619
+ params=LegacyCommandParams.model_construct(
620
+ legacyCommandType=command["name"],
621
+ legacyCommandText=command["payload"]["text"],
622
+ ),
623
+ )
624
+ custom_create = pe_commands.CustomCreate.model_construct(
625
+ key=custom_running.key, params=custom_running.params
626
+ )
627
+ return custom_create, custom_running
628
+
629
+ def _map_labware_load(
630
+ self, labware_load_info: LegacyLabwareLoadInfo
631
+ ) -> List[pe_actions.Action]:
632
+ """Map a legacy labware load to a ProtocolEngine command."""
633
+ now = ModelUtils.get_timestamp()
634
+ count = self._command_count["LOAD_LABWARE"]
635
+ slot = labware_load_info.deck_slot
636
+ location: pe_types.LabwareLocation
637
+ location_sequence: pe_types.LabwareLocationSequence = []
638
+ if labware_load_info.on_module:
639
+ module_id = self._module_id_by_slot[slot]
640
+ location = pe_types.ModuleLocation.model_construct(moduleId=module_id)
641
+ location_sequence.append(
642
+ pe_types.OnModuleLocationSequenceComponent(moduleId=module_id)
643
+ )
644
+ else:
645
+ location = pe_types.DeckSlotLocation.model_construct(slotName=slot)
646
+
647
+ location_sequence.append(
648
+ pe_types.OnAddressableAreaLocationSequenceComponent(
649
+ addressableAreaName=slot.value
650
+ )
651
+ )
652
+ command_id = f"commands.LOAD_LABWARE-{count}"
653
+ labware_id = f"labware-{count}"
654
+ succeeded_command = pe_commands.LoadLabware(
655
+ id=command_id,
656
+ key=command_id,
657
+ status=pe_commands.CommandStatus.SUCCEEDED,
658
+ createdAt=now,
659
+ startedAt=now,
660
+ completedAt=now,
661
+ params=pe_commands.LoadLabwareParams.model_construct(
662
+ location=location,
663
+ loadName=labware_load_info.labware_load_name,
664
+ namespace=labware_load_info.labware_namespace,
665
+ version=labware_load_info.labware_version,
666
+ displayName=labware_load_info.labware_display_name,
667
+ ),
668
+ notes=[],
669
+ result=pe_commands.LoadLabwareResult.model_construct(
670
+ labwareId=labware_id,
671
+ definition=labware_definition_type_adapter.validate_python(
672
+ labware_load_info.labware_definition
673
+ ),
674
+ offsetId=labware_load_info.offset_id,
675
+ locationSequence=location_sequence,
676
+ ),
677
+ )
678
+ queue_action = pe_actions.QueueCommandAction(
679
+ command_id=succeeded_command.id,
680
+ created_at=succeeded_command.createdAt,
681
+ request=pe_commands.LoadLabwareCreate.model_construct(
682
+ key=succeeded_command.key, params=succeeded_command.params
683
+ ),
684
+ request_hash=None,
685
+ )
686
+ run_action = pe_actions.RunCommandAction(
687
+ command_id=succeeded_command.id,
688
+ # We just set this above, so we know it's not None.
689
+ started_at=succeeded_command.startedAt, # type: ignore[arg-type]
690
+ )
691
+ state_update = StateUpdate()
692
+ assert succeeded_command.result is not None
693
+ state_update.set_loaded_labware(
694
+ labware_id=labware_id,
695
+ definition=succeeded_command.result.definition,
696
+ display_name=labware_load_info.labware_display_name,
697
+ offset_id=labware_load_info.offset_id,
698
+ location=location,
699
+ )
700
+
701
+ succeed_action = pe_actions.SucceedCommandAction(
702
+ command=succeeded_command,
703
+ state_update=state_update,
704
+ )
705
+
706
+ self._command_count["LOAD_LABWARE"] = count + 1
707
+ if isinstance(location, pe_types.DeckSlotLocation):
708
+ self._labware_id_by_slot[location.slotName] = labware_id
709
+ elif isinstance(location, pe_types.ModuleLocation):
710
+ self._labware_id_by_module_id[location.moduleId] = labware_id
711
+
712
+ return [queue_action, run_action, succeed_action]
713
+
714
+ def _map_instrument_load(
715
+ self,
716
+ instrument_load_info: LegacyInstrumentLoadInfo,
717
+ ) -> List[pe_actions.Action]:
718
+ """Map a legacy instrument (pipette) load to a ProtocolEngine command.
719
+
720
+ Also creates a `AddPipetteConfigAction`, which is not necessary for the run,
721
+ but is needed for stop so tip geometry is in state for the HardwareStopper.
722
+ """
723
+ now = ModelUtils.get_timestamp()
724
+ count = self._command_count["LOAD_PIPETTE"]
725
+ command_id = f"commands.LOAD_PIPETTE-{count}"
726
+ pipette_id = f"pipette-{count}"
727
+ mount = MountType(str(instrument_load_info.mount).lower())
728
+
729
+ succeeded_command = pe_commands.LoadPipette.model_construct(
730
+ id=command_id,
731
+ key=command_id,
732
+ status=pe_commands.CommandStatus.SUCCEEDED,
733
+ createdAt=now,
734
+ startedAt=now,
735
+ completedAt=now,
736
+ params=pe_commands.LoadPipetteParams.model_construct(
737
+ pipetteName=PipetteNameType(instrument_load_info.instrument_load_name),
738
+ mount=mount,
739
+ ),
740
+ notes=[],
741
+ result=pe_commands.LoadPipetteResult.model_construct(pipetteId=pipette_id),
742
+ )
743
+ serial = instrument_load_info.pipette_dict.get("pipette_id", None) or ""
744
+ state_update = StateUpdate()
745
+ state_update.set_load_pipette(
746
+ pipette_id=pipette_id,
747
+ mount=succeeded_command.params.mount,
748
+ pipette_name=succeeded_command.params.pipetteName,
749
+ liquid_presence_detection=succeeded_command.params.liquidPresenceDetection,
750
+ )
751
+ state_update.update_pipette_config(
752
+ pipette_id=pipette_id,
753
+ serial_number=serial,
754
+ config=pipette_data_provider.get_pipette_static_config(
755
+ # Compatibility note - this is the version of tip overlap data, it stays at 0
756
+ # so protocol behavior does not change when you run a legacy JSON protocol
757
+ instrument_load_info.pipette_dict,
758
+ "v0",
759
+ ),
760
+ )
761
+ queue_action = pe_actions.QueueCommandAction(
762
+ command_id=succeeded_command.id,
763
+ created_at=succeeded_command.createdAt,
764
+ request=pe_commands.LoadPipetteCreate.model_construct(
765
+ key=succeeded_command.key, params=succeeded_command.params
766
+ ),
767
+ request_hash=None,
768
+ )
769
+ run_action = pe_actions.RunCommandAction(
770
+ command_id=succeeded_command.id,
771
+ # We just set this above, so we know it's not None.
772
+ started_at=succeeded_command.startedAt, # type: ignore[arg-type]
773
+ )
774
+
775
+ succeed_action = pe_actions.SucceedCommandAction(
776
+ command=succeeded_command,
777
+ state_update=state_update,
778
+ )
779
+
780
+ self._command_count["LOAD_PIPETTE"] = count + 1
781
+ self._pipette_id_by_mount[mount] = pipette_id
782
+
783
+ return [queue_action, run_action, succeed_action]
784
+
785
+ def _map_module_load(
786
+ self, module_load_info: LegacyModuleLoadInfo
787
+ ) -> List[pe_actions.Action]:
788
+ """Map a legacy module load to a Protocol Engine command."""
789
+ now = ModelUtils.get_timestamp()
790
+
791
+ count = self._command_count["LOAD_MODULE"]
792
+ command_id = f"commands.LOAD_MODULE-{count}"
793
+ module_id = f"module-{count}"
794
+ requested_model = _HARDWARE_TO_PE_MODULE[module_load_info.requested_model]
795
+ loaded_model = _HARDWARE_TO_PE_MODULE[module_load_info.loaded_model]
796
+
797
+ # This will fetch a V2 definition only. PAPI < v2.3 use V1 definitions.
798
+ # When running a < v2.3 protocol, there will be a mismatch of definitions used
799
+ # during analysis+LPC (V2) and protocol execution (V1).
800
+ # But this shouldn't result in any problems since V2 and V1 definitions
801
+ # have similar info, with V2 having additional info fields.
802
+ loaded_definition = self._module_definition_by_model.get(
803
+ loaded_model
804
+ ) or self._module_data_provider.get_definition(loaded_model)
805
+
806
+ succeeded_command = pe_commands.LoadModule.model_construct(
807
+ id=command_id,
808
+ key=command_id,
809
+ status=pe_commands.CommandStatus.SUCCEEDED,
810
+ createdAt=now,
811
+ startedAt=now,
812
+ completedAt=now,
813
+ params=pe_commands.LoadModuleParams.model_construct(
814
+ model=requested_model,
815
+ location=pe_types.DeckSlotLocation(
816
+ slotName=module_load_info.deck_slot,
817
+ ),
818
+ moduleId=module_id,
819
+ ),
820
+ notes=[],
821
+ result=pe_commands.LoadModuleResult.model_construct(
822
+ moduleId=module_id,
823
+ serialNumber=module_load_info.module_serial,
824
+ model=loaded_model,
825
+ ),
826
+ )
827
+ queue_action = pe_actions.QueueCommandAction(
828
+ command_id=succeeded_command.id,
829
+ created_at=succeeded_command.createdAt,
830
+ request=pe_commands.LoadModuleCreate.model_construct(
831
+ key=succeeded_command.key, params=succeeded_command.params
832
+ ),
833
+ request_hash=None,
834
+ )
835
+ run_action = pe_actions.RunCommandAction(
836
+ command_id=succeeded_command.id,
837
+ # We just set this above, so we know it's not None.
838
+ started_at=succeeded_command.startedAt, # type: ignore[arg-type]
839
+ )
840
+ succeed_action = pe_actions.SucceedCommandAction(
841
+ command=succeeded_command, state_update=StateUpdate()
842
+ )
843
+
844
+ self._command_count["LOAD_MODULE"] = count + 1
845
+ self._module_id_by_slot[module_load_info.deck_slot] = module_id
846
+ self._module_definition_by_model[loaded_model] = loaded_definition
847
+
848
+ return [queue_action, run_action, succeed_action]