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
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())