opentrons 8.2.0a3__py2.py3-none-any.whl → 8.3.0__py2.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.
Files changed (238) hide show
  1. opentrons/calibration_storage/deck_configuration.py +3 -3
  2. opentrons/calibration_storage/file_operators.py +3 -3
  3. opentrons/calibration_storage/helpers.py +3 -1
  4. opentrons/calibration_storage/ot2/models/v1.py +16 -29
  5. opentrons/calibration_storage/ot2/tip_length.py +7 -4
  6. opentrons/calibration_storage/ot3/models/v1.py +14 -23
  7. opentrons/cli/analyze.py +18 -6
  8. opentrons/config/defaults_ot3.py +1 -0
  9. opentrons/drivers/asyncio/communication/__init__.py +2 -0
  10. opentrons/drivers/asyncio/communication/errors.py +16 -3
  11. opentrons/drivers/asyncio/communication/serial_connection.py +24 -9
  12. opentrons/drivers/command_builder.py +2 -2
  13. opentrons/drivers/flex_stacker/__init__.py +9 -0
  14. opentrons/drivers/flex_stacker/abstract.py +89 -0
  15. opentrons/drivers/flex_stacker/driver.py +260 -0
  16. opentrons/drivers/flex_stacker/simulator.py +109 -0
  17. opentrons/drivers/flex_stacker/types.py +138 -0
  18. opentrons/drivers/heater_shaker/driver.py +18 -3
  19. opentrons/drivers/temp_deck/driver.py +13 -3
  20. opentrons/drivers/thermocycler/driver.py +17 -3
  21. opentrons/execute.py +3 -1
  22. opentrons/hardware_control/__init__.py +1 -2
  23. opentrons/hardware_control/api.py +33 -21
  24. opentrons/hardware_control/backends/flex_protocol.py +17 -7
  25. opentrons/hardware_control/backends/ot3controller.py +213 -63
  26. opentrons/hardware_control/backends/ot3simulator.py +18 -9
  27. opentrons/hardware_control/backends/ot3utils.py +43 -15
  28. opentrons/hardware_control/dev_types.py +4 -0
  29. opentrons/hardware_control/emulation/heater_shaker.py +4 -0
  30. opentrons/hardware_control/emulation/module_server/client.py +1 -1
  31. opentrons/hardware_control/emulation/module_server/server.py +5 -3
  32. opentrons/hardware_control/emulation/settings.py +3 -4
  33. opentrons/hardware_control/instruments/ot2/instrument_calibration.py +2 -1
  34. opentrons/hardware_control/instruments/ot2/pipette.py +15 -22
  35. opentrons/hardware_control/instruments/ot2/pipette_handler.py +8 -1
  36. opentrons/hardware_control/instruments/ot3/gripper.py +2 -2
  37. opentrons/hardware_control/instruments/ot3/pipette.py +23 -22
  38. opentrons/hardware_control/instruments/ot3/pipette_handler.py +10 -1
  39. opentrons/hardware_control/modules/mod_abc.py +2 -2
  40. opentrons/hardware_control/motion_utilities.py +68 -0
  41. opentrons/hardware_control/nozzle_manager.py +39 -41
  42. opentrons/hardware_control/ot3_calibration.py +1 -1
  43. opentrons/hardware_control/ot3api.py +78 -31
  44. opentrons/hardware_control/protocols/gripper_controller.py +3 -0
  45. opentrons/hardware_control/protocols/hardware_manager.py +5 -1
  46. opentrons/hardware_control/protocols/liquid_handler.py +22 -1
  47. opentrons/hardware_control/protocols/motion_controller.py +7 -0
  48. opentrons/hardware_control/robot_calibration.py +1 -1
  49. opentrons/hardware_control/types.py +61 -0
  50. opentrons/legacy_commands/commands.py +37 -0
  51. opentrons/legacy_commands/types.py +39 -0
  52. opentrons/protocol_api/__init__.py +20 -1
  53. opentrons/protocol_api/_liquid.py +24 -49
  54. opentrons/protocol_api/_liquid_properties.py +754 -0
  55. opentrons/protocol_api/_types.py +24 -0
  56. opentrons/protocol_api/core/common.py +2 -0
  57. opentrons/protocol_api/core/engine/instrument.py +191 -10
  58. opentrons/protocol_api/core/engine/labware.py +29 -7
  59. opentrons/protocol_api/core/engine/protocol.py +130 -5
  60. opentrons/protocol_api/core/engine/robot.py +139 -0
  61. opentrons/protocol_api/core/engine/well.py +4 -1
  62. opentrons/protocol_api/core/instrument.py +73 -4
  63. opentrons/protocol_api/core/labware.py +13 -4
  64. opentrons/protocol_api/core/legacy/legacy_instrument_core.py +87 -3
  65. opentrons/protocol_api/core/legacy/legacy_labware_core.py +13 -4
  66. opentrons/protocol_api/core/legacy/legacy_protocol_core.py +32 -1
  67. opentrons/protocol_api/core/legacy/legacy_robot_core.py +0 -0
  68. opentrons/protocol_api/core/legacy_simulator/legacy_instrument_core.py +61 -3
  69. opentrons/protocol_api/core/protocol.py +34 -1
  70. opentrons/protocol_api/core/robot.py +51 -0
  71. opentrons/protocol_api/instrument_context.py +299 -44
  72. opentrons/protocol_api/labware.py +248 -9
  73. opentrons/protocol_api/module_contexts.py +21 -17
  74. opentrons/protocol_api/protocol_context.py +125 -4
  75. opentrons/protocol_api/robot_context.py +204 -32
  76. opentrons/protocol_api/validation.py +262 -3
  77. opentrons/protocol_engine/__init__.py +4 -0
  78. opentrons/protocol_engine/actions/actions.py +2 -3
  79. opentrons/protocol_engine/clients/sync_client.py +18 -0
  80. opentrons/protocol_engine/commands/__init__.py +121 -0
  81. opentrons/protocol_engine/commands/absorbance_reader/close_lid.py +1 -3
  82. opentrons/protocol_engine/commands/absorbance_reader/initialize.py +20 -6
  83. opentrons/protocol_engine/commands/absorbance_reader/open_lid.py +1 -2
  84. opentrons/protocol_engine/commands/absorbance_reader/read.py +40 -10
  85. opentrons/protocol_engine/commands/air_gap_in_place.py +160 -0
  86. opentrons/protocol_engine/commands/aspirate.py +103 -53
  87. opentrons/protocol_engine/commands/aspirate_in_place.py +55 -51
  88. opentrons/protocol_engine/commands/blow_out.py +44 -39
  89. opentrons/protocol_engine/commands/blow_out_in_place.py +21 -32
  90. opentrons/protocol_engine/commands/calibration/calibrate_gripper.py +13 -6
  91. opentrons/protocol_engine/commands/calibration/calibrate_module.py +1 -1
  92. opentrons/protocol_engine/commands/calibration/calibrate_pipette.py +3 -3
  93. opentrons/protocol_engine/commands/calibration/move_to_maintenance_position.py +1 -1
  94. opentrons/protocol_engine/commands/command.py +73 -66
  95. opentrons/protocol_engine/commands/command_unions.py +140 -1
  96. opentrons/protocol_engine/commands/comment.py +1 -1
  97. opentrons/protocol_engine/commands/configure_for_volume.py +10 -3
  98. opentrons/protocol_engine/commands/configure_nozzle_layout.py +6 -4
  99. opentrons/protocol_engine/commands/custom.py +6 -12
  100. opentrons/protocol_engine/commands/dispense.py +82 -48
  101. opentrons/protocol_engine/commands/dispense_in_place.py +71 -51
  102. opentrons/protocol_engine/commands/drop_tip.py +52 -31
  103. opentrons/protocol_engine/commands/drop_tip_in_place.py +79 -8
  104. opentrons/protocol_engine/commands/evotip_dispense.py +156 -0
  105. opentrons/protocol_engine/commands/evotip_seal_pipette.py +331 -0
  106. opentrons/protocol_engine/commands/evotip_unseal_pipette.py +160 -0
  107. opentrons/protocol_engine/commands/generate_command_schema.py +4 -11
  108. opentrons/protocol_engine/commands/get_next_tip.py +134 -0
  109. opentrons/protocol_engine/commands/get_tip_presence.py +1 -1
  110. opentrons/protocol_engine/commands/heater_shaker/close_labware_latch.py +1 -1
  111. opentrons/protocol_engine/commands/heater_shaker/deactivate_heater.py +1 -1
  112. opentrons/protocol_engine/commands/heater_shaker/deactivate_shaker.py +1 -1
  113. opentrons/protocol_engine/commands/heater_shaker/open_labware_latch.py +1 -1
  114. opentrons/protocol_engine/commands/heater_shaker/set_and_wait_for_shake_speed.py +1 -1
  115. opentrons/protocol_engine/commands/heater_shaker/set_target_temperature.py +1 -1
  116. opentrons/protocol_engine/commands/heater_shaker/wait_for_temperature.py +10 -4
  117. opentrons/protocol_engine/commands/home.py +13 -4
  118. opentrons/protocol_engine/commands/liquid_probe.py +125 -31
  119. opentrons/protocol_engine/commands/load_labware.py +33 -6
  120. opentrons/protocol_engine/commands/load_lid.py +146 -0
  121. opentrons/protocol_engine/commands/load_lid_stack.py +189 -0
  122. opentrons/protocol_engine/commands/load_liquid.py +12 -4
  123. opentrons/protocol_engine/commands/load_liquid_class.py +144 -0
  124. opentrons/protocol_engine/commands/load_module.py +31 -10
  125. opentrons/protocol_engine/commands/load_pipette.py +19 -8
  126. opentrons/protocol_engine/commands/magnetic_module/disengage.py +1 -1
  127. opentrons/protocol_engine/commands/magnetic_module/engage.py +1 -1
  128. opentrons/protocol_engine/commands/move_labware.py +28 -6
  129. opentrons/protocol_engine/commands/move_relative.py +35 -25
  130. opentrons/protocol_engine/commands/move_to_addressable_area.py +40 -27
  131. opentrons/protocol_engine/commands/move_to_addressable_area_for_drop_tip.py +53 -32
  132. opentrons/protocol_engine/commands/move_to_coordinates.py +36 -22
  133. opentrons/protocol_engine/commands/move_to_well.py +40 -24
  134. opentrons/protocol_engine/commands/movement_common.py +338 -0
  135. opentrons/protocol_engine/commands/pick_up_tip.py +49 -27
  136. opentrons/protocol_engine/commands/pipetting_common.py +169 -87
  137. opentrons/protocol_engine/commands/prepare_to_aspirate.py +24 -33
  138. opentrons/protocol_engine/commands/reload_labware.py +1 -1
  139. opentrons/protocol_engine/commands/retract_axis.py +1 -1
  140. opentrons/protocol_engine/commands/robot/__init__.py +69 -0
  141. opentrons/protocol_engine/commands/robot/close_gripper_jaw.py +86 -0
  142. opentrons/protocol_engine/commands/robot/common.py +18 -0
  143. opentrons/protocol_engine/commands/robot/move_axes_relative.py +101 -0
  144. opentrons/protocol_engine/commands/robot/move_axes_to.py +100 -0
  145. opentrons/protocol_engine/commands/robot/move_to.py +94 -0
  146. opentrons/protocol_engine/commands/robot/open_gripper_jaw.py +77 -0
  147. opentrons/protocol_engine/commands/save_position.py +14 -5
  148. opentrons/protocol_engine/commands/set_rail_lights.py +1 -1
  149. opentrons/protocol_engine/commands/set_status_bar.py +1 -1
  150. opentrons/protocol_engine/commands/temperature_module/deactivate.py +1 -1
  151. opentrons/protocol_engine/commands/temperature_module/set_target_temperature.py +1 -1
  152. opentrons/protocol_engine/commands/temperature_module/wait_for_temperature.py +10 -4
  153. opentrons/protocol_engine/commands/thermocycler/close_lid.py +1 -1
  154. opentrons/protocol_engine/commands/thermocycler/deactivate_block.py +1 -1
  155. opentrons/protocol_engine/commands/thermocycler/deactivate_lid.py +1 -1
  156. opentrons/protocol_engine/commands/thermocycler/open_lid.py +1 -1
  157. opentrons/protocol_engine/commands/thermocycler/run_extended_profile.py +9 -3
  158. opentrons/protocol_engine/commands/thermocycler/run_profile.py +9 -3
  159. opentrons/protocol_engine/commands/thermocycler/set_target_block_temperature.py +11 -4
  160. opentrons/protocol_engine/commands/thermocycler/set_target_lid_temperature.py +1 -1
  161. opentrons/protocol_engine/commands/thermocycler/wait_for_block_temperature.py +1 -1
  162. opentrons/protocol_engine/commands/thermocycler/wait_for_lid_temperature.py +1 -1
  163. opentrons/protocol_engine/commands/touch_tip.py +65 -16
  164. opentrons/protocol_engine/commands/unsafe/unsafe_blow_out_in_place.py +5 -2
  165. opentrons/protocol_engine/commands/unsafe/unsafe_drop_tip_in_place.py +13 -4
  166. opentrons/protocol_engine/commands/unsafe/unsafe_engage_axes.py +2 -5
  167. opentrons/protocol_engine/commands/unsafe/unsafe_place_labware.py +1 -1
  168. opentrons/protocol_engine/commands/unsafe/unsafe_ungrip_labware.py +4 -2
  169. opentrons/protocol_engine/commands/unsafe/update_position_estimators.py +2 -5
  170. opentrons/protocol_engine/commands/verify_tip_presence.py +11 -4
  171. opentrons/protocol_engine/commands/wait_for_duration.py +10 -3
  172. opentrons/protocol_engine/commands/wait_for_resume.py +10 -3
  173. opentrons/protocol_engine/errors/__init__.py +12 -0
  174. opentrons/protocol_engine/errors/error_occurrence.py +19 -20
  175. opentrons/protocol_engine/errors/exceptions.py +76 -0
  176. opentrons/protocol_engine/execution/command_executor.py +1 -1
  177. opentrons/protocol_engine/execution/equipment.py +73 -5
  178. opentrons/protocol_engine/execution/gantry_mover.py +369 -8
  179. opentrons/protocol_engine/execution/hardware_stopper.py +7 -7
  180. opentrons/protocol_engine/execution/movement.py +27 -0
  181. opentrons/protocol_engine/execution/pipetting.py +5 -1
  182. opentrons/protocol_engine/execution/tip_handler.py +34 -15
  183. opentrons/protocol_engine/notes/notes.py +1 -1
  184. opentrons/protocol_engine/protocol_engine.py +7 -6
  185. opentrons/protocol_engine/resources/labware_data_provider.py +1 -1
  186. opentrons/protocol_engine/resources/labware_validation.py +18 -0
  187. opentrons/protocol_engine/resources/module_data_provider.py +1 -1
  188. opentrons/protocol_engine/resources/pipette_data_provider.py +26 -0
  189. opentrons/protocol_engine/slot_standardization.py +9 -9
  190. opentrons/protocol_engine/state/_move_types.py +9 -5
  191. opentrons/protocol_engine/state/_well_math.py +193 -0
  192. opentrons/protocol_engine/state/addressable_areas.py +25 -61
  193. opentrons/protocol_engine/state/command_history.py +12 -0
  194. opentrons/protocol_engine/state/commands.py +22 -14
  195. opentrons/protocol_engine/state/files.py +10 -12
  196. opentrons/protocol_engine/state/fluid_stack.py +138 -0
  197. opentrons/protocol_engine/state/frustum_helpers.py +63 -69
  198. opentrons/protocol_engine/state/geometry.py +47 -1
  199. opentrons/protocol_engine/state/labware.py +92 -26
  200. opentrons/protocol_engine/state/liquid_classes.py +82 -0
  201. opentrons/protocol_engine/state/liquids.py +16 -4
  202. opentrons/protocol_engine/state/modules.py +56 -71
  203. opentrons/protocol_engine/state/motion.py +6 -1
  204. opentrons/protocol_engine/state/pipettes.py +149 -58
  205. opentrons/protocol_engine/state/state.py +21 -2
  206. opentrons/protocol_engine/state/state_summary.py +4 -2
  207. opentrons/protocol_engine/state/tips.py +11 -44
  208. opentrons/protocol_engine/state/update_types.py +343 -48
  209. opentrons/protocol_engine/state/wells.py +19 -11
  210. opentrons/protocol_engine/types.py +176 -28
  211. opentrons/protocol_reader/extract_labware_definitions.py +5 -2
  212. opentrons/protocol_reader/file_format_validator.py +5 -5
  213. opentrons/protocol_runner/json_file_reader.py +9 -3
  214. opentrons/protocol_runner/json_translator.py +51 -25
  215. opentrons/protocol_runner/legacy_command_mapper.py +66 -64
  216. opentrons/protocol_runner/protocol_runner.py +35 -4
  217. opentrons/protocol_runner/python_protocol_wrappers.py +1 -1
  218. opentrons/protocol_runner/run_orchestrator.py +13 -3
  219. opentrons/protocols/advanced_control/common.py +38 -0
  220. opentrons/protocols/advanced_control/mix.py +1 -1
  221. opentrons/protocols/advanced_control/transfers/__init__.py +0 -0
  222. opentrons/protocols/advanced_control/transfers/common.py +56 -0
  223. opentrons/protocols/advanced_control/{transfers.py → transfers/transfer.py} +10 -85
  224. opentrons/protocols/api_support/definitions.py +1 -1
  225. opentrons/protocols/api_support/instrument.py +1 -1
  226. opentrons/protocols/api_support/util.py +10 -0
  227. opentrons/protocols/labware.py +70 -8
  228. opentrons/protocols/models/json_protocol.py +5 -9
  229. opentrons/simulate.py +3 -1
  230. opentrons/types.py +162 -2
  231. opentrons/util/entrypoint_util.py +2 -5
  232. opentrons/util/logging_config.py +1 -1
  233. {opentrons-8.2.0a3.dist-info → opentrons-8.3.0.dist-info}/METADATA +16 -15
  234. {opentrons-8.2.0a3.dist-info → opentrons-8.3.0.dist-info}/RECORD +238 -208
  235. {opentrons-8.2.0a3.dist-info → opentrons-8.3.0.dist-info}/WHEEL +1 -1
  236. {opentrons-8.2.0a3.dist-info → opentrons-8.3.0.dist-info}/LICENSE +0 -0
  237. {opentrons-8.2.0a3.dist-info → opentrons-8.3.0.dist-info}/entry_points.txt +0 -0
  238. {opentrons-8.2.0a3.dist-info → opentrons-8.3.0.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,146 @@
1
+ """Load lid command request, result, and implementation models."""
2
+ from __future__ import annotations
3
+ from pydantic import BaseModel, Field
4
+ from typing import TYPE_CHECKING, Optional, Type
5
+ from typing_extensions import Literal
6
+
7
+ from opentrons_shared_data.labware.labware_definition import LabwareDefinition
8
+
9
+ from ..errors import LabwareCannotBeStackedError, LabwareIsNotAllowedInLocationError
10
+ from ..resources import labware_validation
11
+ from ..types import (
12
+ LabwareLocation,
13
+ OnLabwareLocation,
14
+ )
15
+
16
+ from .command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData
17
+ from ..errors.error_occurrence import ErrorOccurrence
18
+ from ..state.update_types import StateUpdate
19
+
20
+ if TYPE_CHECKING:
21
+ from ..state.state import StateView
22
+ from ..execution import EquipmentHandler
23
+
24
+
25
+ LoadLidCommandType = Literal["loadLid"]
26
+
27
+
28
+ class LoadLidParams(BaseModel):
29
+ """Payload required to load a lid onto a labware."""
30
+
31
+ location: LabwareLocation = Field(
32
+ ...,
33
+ description="Labware the lid should be loaded onto.",
34
+ )
35
+ loadName: str = Field(
36
+ ...,
37
+ description="Name used to reference a lid labware definition.",
38
+ )
39
+ namespace: str = Field(
40
+ ...,
41
+ description="The namespace the lid labware definition belongs to.",
42
+ )
43
+ version: int = Field(
44
+ ...,
45
+ description="The lid labware definition version.",
46
+ )
47
+
48
+
49
+ class LoadLidResult(BaseModel):
50
+ """Result data from the execution of a LoadLabware command."""
51
+
52
+ labwareId: str = Field(
53
+ ...,
54
+ description="An ID to reference this lid labware in subsequent commands.",
55
+ )
56
+ definition: LabwareDefinition = Field(
57
+ ...,
58
+ description="The full definition data for this lid labware.",
59
+ )
60
+
61
+
62
+ class LoadLidImplementation(
63
+ AbstractCommandImpl[LoadLidParams, SuccessData[LoadLidResult]]
64
+ ):
65
+ """Load lid command implementation."""
66
+
67
+ def __init__(
68
+ self, equipment: EquipmentHandler, state_view: StateView, **kwargs: object
69
+ ) -> None:
70
+ self._equipment = equipment
71
+ self._state_view = state_view
72
+
73
+ async def execute(self, params: LoadLidParams) -> SuccessData[LoadLidResult]:
74
+ """Load definition and calibration data necessary for a lid."""
75
+ if not isinstance(params.location, OnLabwareLocation):
76
+ raise LabwareIsNotAllowedInLocationError(
77
+ "Lid Labware is only allowed to be loaded on top of a labware. Try `load_lid_stack(...)` to load lids without parent labware."
78
+ )
79
+
80
+ verified_location = self._state_view.geometry.ensure_location_not_occupied(
81
+ params.location
82
+ )
83
+ loaded_labware = await self._equipment.load_labware(
84
+ load_name=params.loadName,
85
+ namespace=params.namespace,
86
+ version=params.version,
87
+ location=verified_location,
88
+ labware_id=None,
89
+ )
90
+
91
+ # TODO(chb 2024-12-12) these validation checks happen after the labware is loaded, because they rely on
92
+ # on the definition. In practice this will not cause any issues since they will raise protocol ending
93
+ # exception, but for correctness should be refactored to do this check beforehand.
94
+ if not labware_validation.validate_definition_is_lid(loaded_labware.definition):
95
+ raise LabwareCannotBeStackedError(
96
+ f"Labware {params.loadName} is not a Lid and cannot be loaded onto {self._state_view.labware.get_display_name(params.location.labwareId)}."
97
+ )
98
+
99
+ state_update = StateUpdate()
100
+
101
+ # In the case of lids being loaded on top of other labware, set the parent labware's lid
102
+ state_update.set_lid(
103
+ parent_labware_id=params.location.labwareId,
104
+ lid_id=loaded_labware.labware_id,
105
+ )
106
+
107
+ state_update.set_loaded_labware(
108
+ labware_id=loaded_labware.labware_id,
109
+ offset_id=loaded_labware.offsetId,
110
+ definition=loaded_labware.definition,
111
+ location=verified_location,
112
+ display_name=None,
113
+ )
114
+
115
+ if isinstance(verified_location, OnLabwareLocation):
116
+ self._state_view.labware.raise_if_labware_cannot_be_stacked(
117
+ top_labware_definition=loaded_labware.definition,
118
+ bottom_labware_id=verified_location.labwareId,
119
+ )
120
+
121
+ return SuccessData(
122
+ public=LoadLidResult(
123
+ labwareId=loaded_labware.labware_id,
124
+ definition=loaded_labware.definition,
125
+ ),
126
+ state_update=state_update,
127
+ )
128
+
129
+
130
+ class LoadLid(BaseCommand[LoadLidParams, LoadLidResult, ErrorOccurrence]):
131
+ """Load lid command resource model."""
132
+
133
+ commandType: LoadLidCommandType = "loadLid"
134
+ params: LoadLidParams
135
+ result: Optional[LoadLidResult] = None
136
+
137
+ _ImplementationCls: Type[LoadLidImplementation] = LoadLidImplementation
138
+
139
+
140
+ class LoadLidCreate(BaseCommandCreate[LoadLidParams]):
141
+ """Load lid command creation request."""
142
+
143
+ commandType: LoadLidCommandType = "loadLid"
144
+ params: LoadLidParams
145
+
146
+ _CommandCls: Type[LoadLid] = LoadLid
@@ -0,0 +1,189 @@
1
+ """Load lid stack command request, result, and implementation models."""
2
+ from __future__ import annotations
3
+ from pydantic import BaseModel, Field
4
+ from typing import TYPE_CHECKING, Optional, Type, List
5
+ from typing_extensions import Literal
6
+
7
+ from opentrons_shared_data.labware.labware_definition import LabwareDefinition
8
+
9
+ from ..errors import LabwareIsNotAllowedInLocationError, ProtocolEngineError
10
+ from ..resources import fixture_validation, labware_validation
11
+ from ..types import (
12
+ LabwareLocation,
13
+ OnLabwareLocation,
14
+ DeckSlotLocation,
15
+ AddressableAreaLocation,
16
+ )
17
+
18
+ from .command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData
19
+ from ..errors.error_occurrence import ErrorOccurrence
20
+ from ..state.update_types import StateUpdate
21
+
22
+ if TYPE_CHECKING:
23
+ from ..state.state import StateView
24
+ from ..execution import EquipmentHandler
25
+
26
+
27
+ LoadLidStackCommandType = Literal["loadLidStack"]
28
+
29
+ _LID_STACK_PE_LABWARE = "protocol_engine_lid_stack_object"
30
+ _LID_STACK_PE_NAMESPACE = "opentrons"
31
+ _LID_STACK_PE_VERSION = 1
32
+
33
+
34
+ class LoadLidStackParams(BaseModel):
35
+ """Payload required to load a lid stack onto a location."""
36
+
37
+ location: LabwareLocation = Field(
38
+ ...,
39
+ description="Location the lid stack should be loaded into.",
40
+ )
41
+ loadName: str = Field(
42
+ ...,
43
+ description="Name used to reference a lid labware definition.",
44
+ )
45
+ namespace: str = Field(
46
+ ...,
47
+ description="The namespace the lid labware definition belongs to.",
48
+ )
49
+ version: int = Field(
50
+ ...,
51
+ description="The lid labware definition version.",
52
+ )
53
+ quantity: int = Field(
54
+ ...,
55
+ description="The quantity of lids to load.",
56
+ )
57
+
58
+
59
+ class LoadLidStackResult(BaseModel):
60
+ """Result data from the execution of a LoadLidStack command."""
61
+
62
+ stackLabwareId: str = Field(
63
+ ...,
64
+ description="An ID to reference the Protocol Engine Labware Lid Stack in subsequent commands.",
65
+ )
66
+ labwareIds: List[str] = Field(
67
+ ...,
68
+ description="A list of lid labware IDs to reference the lids in this stack by. The first ID is the bottom of the stack.",
69
+ )
70
+ definition: LabwareDefinition = Field(
71
+ ...,
72
+ description="The full definition data for this lid labware.",
73
+ )
74
+ location: LabwareLocation = Field(
75
+ ..., description="The Location that the stack of lid labware has been loaded."
76
+ )
77
+
78
+
79
+ class LoadLidStackImplementation(
80
+ AbstractCommandImpl[LoadLidStackParams, SuccessData[LoadLidStackResult]]
81
+ ):
82
+ """Load lid stack command implementation."""
83
+
84
+ def __init__(
85
+ self, equipment: EquipmentHandler, state_view: StateView, **kwargs: object
86
+ ) -> None:
87
+ self._equipment = equipment
88
+ self._state_view = state_view
89
+
90
+ async def execute(
91
+ self, params: LoadLidStackParams
92
+ ) -> SuccessData[LoadLidStackResult]:
93
+ """Load definition and calibration data necessary for a lid stack."""
94
+ if isinstance(params.location, AddressableAreaLocation):
95
+ area_name = params.location.addressableAreaName
96
+ if not (
97
+ fixture_validation.is_deck_slot(params.location.addressableAreaName)
98
+ or fixture_validation.is_abs_reader(params.location.addressableAreaName)
99
+ ):
100
+ raise LabwareIsNotAllowedInLocationError(
101
+ f"Cannot load {params.loadName} onto addressable area {area_name}"
102
+ )
103
+ self._state_view.addressable_areas.raise_if_area_not_in_deck_configuration(
104
+ area_name
105
+ )
106
+ elif isinstance(params.location, DeckSlotLocation):
107
+ self._state_view.addressable_areas.raise_if_area_not_in_deck_configuration(
108
+ params.location.slotName.id
109
+ )
110
+
111
+ verified_location = self._state_view.geometry.ensure_location_not_occupied(
112
+ params.location
113
+ )
114
+
115
+ lid_stack_object = await self._equipment.load_labware(
116
+ load_name=_LID_STACK_PE_LABWARE,
117
+ namespace=_LID_STACK_PE_NAMESPACE,
118
+ version=_LID_STACK_PE_VERSION,
119
+ location=verified_location,
120
+ labware_id=None,
121
+ )
122
+ if not labware_validation.validate_definition_is_system(
123
+ lid_stack_object.definition
124
+ ):
125
+ raise ProtocolEngineError(
126
+ message="Lid Stack Labware Object Labware Definition does not contain required allowed role 'system'."
127
+ )
128
+
129
+ loaded_lid_labwares = await self._equipment.load_lids(
130
+ load_name=params.loadName,
131
+ namespace=params.namespace,
132
+ version=params.version,
133
+ location=OnLabwareLocation(labwareId=lid_stack_object.labware_id),
134
+ quantity=params.quantity,
135
+ )
136
+ loaded_lid_locations_by_id = {}
137
+ load_location = OnLabwareLocation(labwareId=lid_stack_object.labware_id)
138
+ for loaded_lid in loaded_lid_labwares:
139
+ loaded_lid_locations_by_id[loaded_lid.labware_id] = load_location
140
+ load_location = OnLabwareLocation(labwareId=loaded_lid.labware_id)
141
+
142
+ state_update = StateUpdate()
143
+ state_update.set_loaded_lid_stack(
144
+ stack_id=lid_stack_object.labware_id,
145
+ stack_object_definition=lid_stack_object.definition,
146
+ stack_location=verified_location,
147
+ labware_ids=list(loaded_lid_locations_by_id.keys()),
148
+ labware_definition=loaded_lid_labwares[0].definition,
149
+ locations=loaded_lid_locations_by_id,
150
+ )
151
+
152
+ if isinstance(verified_location, OnLabwareLocation):
153
+ self._state_view.labware.raise_if_labware_cannot_be_stacked(
154
+ top_labware_definition=loaded_lid_labwares[
155
+ params.quantity - 1
156
+ ].definition,
157
+ bottom_labware_id=verified_location.labwareId,
158
+ )
159
+
160
+ return SuccessData(
161
+ public=LoadLidStackResult(
162
+ stackLabwareId=lid_stack_object.labware_id,
163
+ labwareIds=list(loaded_lid_locations_by_id.keys()),
164
+ definition=loaded_lid_labwares[0].definition,
165
+ location=params.location,
166
+ ),
167
+ state_update=state_update,
168
+ )
169
+
170
+
171
+ class LoadLidStack(
172
+ BaseCommand[LoadLidStackParams, LoadLidStackResult, ErrorOccurrence]
173
+ ):
174
+ """Load lid stack command resource model."""
175
+
176
+ commandType: LoadLidStackCommandType = "loadLidStack"
177
+ params: LoadLidStackParams
178
+ result: Optional[LoadLidStackResult] = None
179
+
180
+ _ImplementationCls: Type[LoadLidStackImplementation] = LoadLidStackImplementation
181
+
182
+
183
+ class LoadLidStackCreate(BaseCommandCreate[LoadLidStackParams]):
184
+ """Load lid stack command creation request."""
185
+
186
+ commandType: LoadLidStackCommandType = "loadLidStack"
187
+ params: LoadLidStackParams
188
+
189
+ _CommandCls: Type[LoadLidStack] = LoadLidStack
@@ -5,6 +5,8 @@ from typing import Optional, Type, Dict, TYPE_CHECKING
5
5
  from typing_extensions import Literal
6
6
 
7
7
  from opentrons.protocol_engine.state.update_types import StateUpdate
8
+ from opentrons.protocol_engine.types import LiquidId
9
+ from opentrons.protocol_engine.errors import InvalidLiquidError
8
10
 
9
11
  from .command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData
10
12
  from ..errors.error_occurrence import ErrorOccurrence
@@ -19,9 +21,9 @@ LoadLiquidCommandType = Literal["loadLiquid"]
19
21
  class LoadLiquidParams(BaseModel):
20
22
  """Payload required to load a liquid into a well."""
21
23
 
22
- liquidId: str = Field(
24
+ liquidId: LiquidId = Field(
23
25
  ...,
24
- description="Unique identifier of the liquid to load.",
26
+ description="Unique identifier of the liquid to load. If this is the sentinel value EMPTY, all values of volumeByWell must be 0.",
25
27
  )
26
28
  labwareId: str = Field(
27
29
  ...,
@@ -29,7 +31,7 @@ class LoadLiquidParams(BaseModel):
29
31
  )
30
32
  volumeByWell: Dict[str, float] = Field(
31
33
  ...,
32
- description="Volume of liquid, in µL, loaded into each well by name, in this labware.",
34
+ description="Volume of liquid, in µL, loaded into each well by name, in this labware. If the liquid id is the sentinel value EMPTY, all volumes must be 0.",
33
35
  )
34
36
 
35
37
 
@@ -57,6 +59,12 @@ class LoadLiquidImplementation(
57
59
  self._state_view.labware.validate_liquid_allowed_in_labware(
58
60
  labware_id=params.labwareId, wells=params.volumeByWell
59
61
  )
62
+ if params.liquidId == "EMPTY":
63
+ for well_name, volume in params.volumeByWell.items():
64
+ if volume != 0.0:
65
+ raise InvalidLiquidError(
66
+ 'loadLiquid commands that specify the special liquid "EMPTY" must set volume to be 0.0, but the volume for {well_name} is {volume}'
67
+ )
60
68
 
61
69
  state_update = StateUpdate()
62
70
  state_update.set_liquid_loaded(
@@ -73,7 +81,7 @@ class LoadLiquid(BaseCommand[LoadLiquidParams, LoadLiquidResult, ErrorOccurrence
73
81
 
74
82
  commandType: LoadLiquidCommandType = "loadLiquid"
75
83
  params: LoadLiquidParams
76
- result: Optional[LoadLiquidResult]
84
+ result: Optional[LoadLiquidResult] = None
77
85
 
78
86
  _ImplementationCls: Type[LoadLiquidImplementation] = LoadLiquidImplementation
79
87
 
@@ -0,0 +1,144 @@
1
+ """LoadLiquidClass stores the liquid class settings used for a transfer into the Protocol Engine."""
2
+ from __future__ import annotations
3
+
4
+ from typing import Optional, Type, TYPE_CHECKING, Any
5
+
6
+ from typing_extensions import Literal
7
+ from pydantic import BaseModel, Field
8
+ from pydantic.json_schema import SkipJsonSchema
9
+
10
+ from .command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData
11
+ from ..errors import LiquidClassDoesNotExistError
12
+ from ..errors.error_occurrence import ErrorOccurrence
13
+ from ..errors.exceptions import LiquidClassRedefinitionError
14
+ from ..state.update_types import LiquidClassLoadedUpdate, StateUpdate
15
+ from ..types import LiquidClassRecord
16
+
17
+ if TYPE_CHECKING:
18
+ from ..state.state import StateView
19
+ from ..resources import ModelUtils
20
+
21
+ LoadLiquidClassCommandType = Literal["loadLiquidClass"]
22
+
23
+
24
+ def _remove_default(s: dict[str, Any]) -> None:
25
+ s.pop("default", None)
26
+
27
+
28
+ class LoadLiquidClassParams(BaseModel):
29
+ """The liquid class transfer properties to store."""
30
+
31
+ liquidClassId: str | SkipJsonSchema[None] = Field(
32
+ None,
33
+ description="Unique identifier for the liquid class to store. "
34
+ "If you do not supply a liquidClassId, we will generate one.",
35
+ json_schema_extra=_remove_default,
36
+ )
37
+ liquidClassRecord: LiquidClassRecord = Field(
38
+ ...,
39
+ description="The liquid class to store.",
40
+ )
41
+
42
+
43
+ class LoadLiquidClassResult(BaseModel):
44
+ """Result from execution of LoadLiquidClass command."""
45
+
46
+ liquidClassId: str = Field(
47
+ ...,
48
+ description="The ID for the liquid class that was loaded, either the one you "
49
+ "supplied or the one we generated.",
50
+ )
51
+
52
+
53
+ class LoadLiquidClassImplementation(
54
+ AbstractCommandImpl[LoadLiquidClassParams, SuccessData[LoadLiquidClassResult]]
55
+ ):
56
+ """Load Liquid Class command implementation."""
57
+
58
+ def __init__(
59
+ self, state_view: StateView, model_utils: ModelUtils, **kwargs: object
60
+ ) -> None:
61
+ self._state_view = state_view
62
+ self._model_utils = model_utils
63
+
64
+ async def execute(
65
+ self, params: LoadLiquidClassParams
66
+ ) -> SuccessData[LoadLiquidClassResult]:
67
+ """Store the liquid class in the Protocol Engine."""
68
+ liquid_class_id: Optional[str]
69
+ already_loaded = False
70
+
71
+ if params.liquidClassId:
72
+ liquid_class_id = params.liquidClassId
73
+ if self._liquid_class_id_already_loaded(
74
+ liquid_class_id, params.liquidClassRecord
75
+ ):
76
+ already_loaded = True
77
+ else:
78
+ liquid_class_id = (
79
+ self._state_view.liquid_classes.get_id_for_liquid_class_record(
80
+ params.liquidClassRecord
81
+ ) # if liquidClassRecord was already loaded, reuse the existing ID
82
+ )
83
+ if liquid_class_id:
84
+ already_loaded = True
85
+ else:
86
+ liquid_class_id = self._model_utils.generate_id()
87
+
88
+ if already_loaded:
89
+ state_update = StateUpdate() # liquid class already loaded, do nothing
90
+ else:
91
+ state_update = StateUpdate(
92
+ liquid_class_loaded=LiquidClassLoadedUpdate(
93
+ liquid_class_id=liquid_class_id,
94
+ liquid_class_record=params.liquidClassRecord,
95
+ )
96
+ )
97
+
98
+ return SuccessData(
99
+ public=LoadLiquidClassResult(liquidClassId=liquid_class_id),
100
+ state_update=state_update,
101
+ )
102
+
103
+ def _liquid_class_id_already_loaded(
104
+ self, liquid_class_id: str, liquid_class_record: LiquidClassRecord
105
+ ) -> bool:
106
+ """Check if the liquid_class_id has already been loaded.
107
+
108
+ If it has, make sure that liquid_class_record matches the previously loaded definition.
109
+ """
110
+ try:
111
+ existing_liquid_class_record = self._state_view.liquid_classes.get(
112
+ liquid_class_id
113
+ )
114
+ except LiquidClassDoesNotExistError:
115
+ return False
116
+
117
+ if liquid_class_record != existing_liquid_class_record:
118
+ raise LiquidClassRedefinitionError(
119
+ f"Liquid class {liquid_class_id} conflicts with previously loaded definition."
120
+ )
121
+ return True
122
+
123
+
124
+ class LoadLiquidClass(
125
+ BaseCommand[LoadLiquidClassParams, LoadLiquidClassResult, ErrorOccurrence]
126
+ ):
127
+ """Load Liquid Class command resource model."""
128
+
129
+ commandType: LoadLiquidClassCommandType = "loadLiquidClass"
130
+ params: LoadLiquidClassParams
131
+ result: Optional[LoadLiquidClassResult] = None
132
+
133
+ _ImplementationCls: Type[
134
+ LoadLiquidClassImplementation
135
+ ] = LoadLiquidClassImplementation
136
+
137
+
138
+ class LoadLiquidClassCreate(BaseCommandCreate[LoadLiquidClassParams]):
139
+ """Load Liquid Class command creation request."""
140
+
141
+ commandType: LoadLiquidClassCommandType = "loadLiquidClass"
142
+ params: LoadLiquidClassParams
143
+
144
+ _CommandCls: Type[LoadLiquidClass] = LoadLiquidClass
@@ -1,8 +1,11 @@
1
1
  """Implementation, request models, and response models for the load module command."""
2
2
  from __future__ import annotations
3
- from typing import TYPE_CHECKING, Optional, Type
3
+ from typing import TYPE_CHECKING, Optional, Type, Any
4
4
  from typing_extensions import Literal
5
5
  from pydantic import BaseModel, Field
6
+ from pydantic.json_schema import SkipJsonSchema
7
+
8
+ from opentrons.protocol_engine.state.update_types import StateUpdate
6
9
 
7
10
  from .command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData
8
11
  from ..errors.error_occurrence import ErrorOccurrence
@@ -25,6 +28,10 @@ if TYPE_CHECKING:
25
28
  LoadModuleCommandType = Literal["loadModule"]
26
29
 
27
30
 
31
+ def _remove_default(s: dict[str, Any]) -> None:
32
+ s.pop("default", None)
33
+
34
+
28
35
  class LoadModuleParams(BaseModel):
29
36
  """Payload required to load a module."""
30
37
 
@@ -57,12 +64,13 @@ class LoadModuleParams(BaseModel):
57
64
  ),
58
65
  )
59
66
 
60
- moduleId: Optional[str] = Field(
67
+ moduleId: str | SkipJsonSchema[None] = Field(
61
68
  None,
62
69
  description=(
63
70
  "An optional ID to assign to this module."
64
71
  " If None, an ID will be generated."
65
72
  ),
73
+ json_schema_extra=_remove_default,
66
74
  )
67
75
 
68
76
 
@@ -75,7 +83,7 @@ class LoadModuleResult(BaseModel):
75
83
 
76
84
  # TODO(mm, 2023-04-13): Remove this field. Jira RSS-221.
77
85
  definition: ModuleDefinition = Field(
78
- deprecated=True,
86
+ json_schema_extra={"deprecated": True},
79
87
  description=(
80
88
  "The definition of the connected module."
81
89
  " This field is an implementation detail. We might change or remove it without warning."
@@ -116,26 +124,35 @@ class LoadModuleImplementation(
116
124
 
117
125
  async def execute(self, params: LoadModuleParams) -> SuccessData[LoadModuleResult]:
118
126
  """Check that the requested module is attached and assign its identifier."""
127
+ state_update = StateUpdate()
128
+
119
129
  module_type = params.model.as_type()
120
130
  self._ensure_module_location(params.location.slotName, module_type)
121
131
 
132
+ # todo(mm, 2024-12-03): Theoretically, we should be able to deal with
133
+ # addressable areas and deck configurations the same way between OT-2 and Flex.
134
+ # Can this be simplified?
122
135
  if self._state_view.config.robot_type == "OT-2 Standard":
123
136
  self._state_view.addressable_areas.raise_if_area_not_in_deck_configuration(
124
137
  params.location.slotName.id
125
138
  )
126
139
  else:
127
- addressable_area = self._state_view.geometry._modules.ensure_and_convert_module_fixture_location(
128
- deck_slot=params.location.slotName,
129
- deck_type=self._state_view.config.deck_type,
130
- model=params.model,
140
+ addressable_area_provided_by_module = (
141
+ self._state_view.modules.ensure_and_convert_module_fixture_location(
142
+ deck_slot=params.location.slotName,
143
+ model=params.model,
144
+ )
131
145
  )
132
146
  self._state_view.addressable_areas.raise_if_area_not_in_deck_configuration(
133
- addressable_area
147
+ addressable_area_provided_by_module
134
148
  )
135
149
 
136
150
  verified_location = self._state_view.geometry.ensure_location_not_occupied(
137
151
  params.location
138
152
  )
153
+ state_update.set_addressable_area_used(
154
+ addressable_area_name=params.location.slotName.id
155
+ )
139
156
 
140
157
  if params.model == ModuleModel.MAGNETIC_BLOCK_V1:
141
158
  loaded_module = await self._equipment.load_magnetic_block(
@@ -157,11 +174,15 @@ class LoadModuleImplementation(
157
174
  model=loaded_module.definition.model,
158
175
  definition=loaded_module.definition,
159
176
  ),
177
+ state_update=state_update,
160
178
  )
161
179
 
162
180
  def _ensure_module_location(
163
181
  self, slot: DeckSlotName, module_type: ModuleType
164
182
  ) -> None:
183
+ # todo(mm, 2024-12-03): Theoretically, we should be able to deal with
184
+ # addressable areas and deck configurations the same way between OT-2 and Flex.
185
+ # Can this be simplified?
165
186
  if self._state_view.config.robot_type == "OT-2 Standard":
166
187
  slot_def = self._state_view.addressable_areas.get_slot_definition(slot.id)
167
188
  compatible_modules = slot_def["compatibleModuleTypes"]
@@ -173,7 +194,7 @@ class LoadModuleImplementation(
173
194
  cutout_fixture_id = ModuleType.to_module_fixture_id(module_type)
174
195
  module_fixture = deck_configuration_provider.get_cutout_fixture(
175
196
  cutout_fixture_id,
176
- self._state_view.addressable_areas.state.deck_definition,
197
+ self._state_view.labware.get_deck_definition(),
177
198
  )
178
199
  cutout_id = (
179
200
  self._state_view.addressable_areas.get_cutout_id_by_deck_slot_name(slot)
@@ -189,7 +210,7 @@ class LoadModule(BaseCommand[LoadModuleParams, LoadModuleResult, ErrorOccurrence
189
210
 
190
211
  commandType: LoadModuleCommandType = "loadModule"
191
212
  params: LoadModuleParams
192
- result: Optional[LoadModuleResult]
213
+ result: Optional[LoadModuleResult] = None
193
214
 
194
215
  _ImplementationCls: Type[LoadModuleImplementation] = LoadModuleImplementation
195
216