1
1
"""Tests for the transfer APIs using liquid classes."""
2
2
import pytest
3
+ import mock
3
4
from decoy import Decoy
4
5
from opentrons_shared_data .robot .types import RobotTypeEnum
5
6
6
7
from opentrons .protocol_api import ProtocolContext
7
8
from opentrons .config import feature_flags as ff
9
+ from opentrons .protocol_api .core .engine import InstrumentCore
10
+ from opentrons .protocol_api .core .engine .transfer_components_executor import (
11
+ TransferType ,
12
+ LiquidAndAirGapPair ,
13
+ )
8
14
9
15
10
16
@pytest .mark .ot3_only
11
17
@pytest .mark .parametrize (
12
18
"simulated_protocol_context" , [("2.20" , "Flex" )], indirect = True
13
19
)
14
- def test_water_transfer (
20
+ def test_water_transfer_with_volume_more_than_tip_max (
15
21
decoy : Decoy , mock_feature_flags : None , simulated_protocol_context : ProtocolContext
16
22
) -> None :
17
23
"""It should run the transfer steps without any errors.
@@ -36,29 +42,194 @@ def test_water_transfer(
36
42
)
37
43
38
44
water = simulated_protocol_context .define_liquid_class ("water" )
39
- pipette_50 .transfer_liquid (
40
- liquid_class = water ,
41
- volume = 60 ,
42
- source = nest_plate .rows ()[0 ],
43
- dest = arma_plate .rows ()[0 ],
44
- new_tip = "always" ,
45
- trash_location = trash ,
45
+ with mock .patch .object (
46
+ InstrumentCore ,
47
+ "pick_up_tip" ,
48
+ side_effect = InstrumentCore .pick_up_tip ,
49
+ autospec = True ,
50
+ ) as patched_pick_up_tip :
51
+ mock_manager = mock .Mock ()
52
+ mock_manager .attach_mock (patched_pick_up_tip , "pick_up_tip" )
53
+
54
+ pipette_50 .transfer_liquid (
55
+ liquid_class = water ,
56
+ volume = 60 ,
57
+ source = nest_plate .rows ()[0 ],
58
+ dest = arma_plate .rows ()[0 ],
59
+ new_tip = "always" ,
60
+ trash_location = trash ,
61
+ )
62
+ assert patched_pick_up_tip .call_count == 24
63
+ patched_pick_up_tip .reset_mock ()
64
+
65
+ pipette_50 .transfer_liquid (
66
+ liquid_class = water ,
67
+ volume = 100 ,
68
+ source = nest_plate .rows ()[0 ],
69
+ dest = arma_plate .rows ()[0 ],
70
+ new_tip = "per source" ,
71
+ trash_location = trash ,
72
+ )
73
+ assert patched_pick_up_tip .call_count == 12
74
+ patched_pick_up_tip .reset_mock ()
75
+
76
+ pipette_50 .pick_up_tip ()
77
+ pipette_50 .transfer_liquid (
78
+ liquid_class = water ,
79
+ volume = 50 ,
80
+ source = nest_plate .rows ()[0 ],
81
+ dest = arma_plate .rows ()[0 ],
82
+ new_tip = "never" ,
83
+ trash_location = trash ,
84
+ )
85
+ pipette_50 .drop_tip ()
86
+ assert patched_pick_up_tip .call_count == 1
87
+
88
+
89
+ @pytest .mark .ot3_only
90
+ @pytest .mark .parametrize (
91
+ "simulated_protocol_context" , [("2.20" , "Flex" )], indirect = True
92
+ )
93
+ def test_order_of_water_transfer_steps (
94
+ decoy : Decoy , mock_feature_flags : None , simulated_protocol_context : ProtocolContext
95
+ ) -> None :
96
+ """It should run the transfer steps without any errors.
97
+
98
+ This test only checks that various supported configurations for a transfer
99
+ analyze successfully. It doesn't check whether the steps are as expected.
100
+ That will be covered in analysis snapshot tests.
101
+ """
102
+ decoy .when (ff .allow_liquid_classes (RobotTypeEnum .FLEX )).then_return (True )
103
+ trash = simulated_protocol_context .load_trash_bin ("A3" )
104
+ tiprack = simulated_protocol_context .load_labware (
105
+ "opentrons_flex_96_tiprack_50ul" , "D1"
46
106
)
47
- pipette_50 .transfer_liquid (
48
- liquid_class = water ,
49
- volume = 60 ,
50
- source = nest_plate .rows ()[0 ],
51
- dest = arma_plate .rows ()[0 ],
52
- new_tip = "per source" ,
53
- trash_location = trash ,
107
+ pipette_50 = simulated_protocol_context .load_instrument (
108
+ "flex_1channel_50" , mount = "left" , tip_racks = [tiprack ]
54
109
)
55
- pipette_50 .pick_up_tip ()
56
- pipette_50 .transfer_liquid (
57
- liquid_class = water ,
58
- volume = 50 ,
59
- source = nest_plate .rows ()[0 ],
60
- dest = arma_plate .rows ()[0 ],
61
- new_tip = "never" ,
62
- trash_location = trash ,
110
+ nest_plate = simulated_protocol_context .load_labware (
111
+ "nest_96_wellplate_200ul_flat" , "C3"
63
112
)
64
- pipette_50 .drop_tip ()
113
+ arma_plate = simulated_protocol_context .load_labware (
114
+ "armadillo_96_wellplate_200ul_pcr_full_skirt" , "C2"
115
+ )
116
+
117
+ water = simulated_protocol_context .define_liquid_class ("water" )
118
+ with (
119
+ mock .patch .object (
120
+ InstrumentCore ,
121
+ "load_liquid_class" ,
122
+ side_effect = InstrumentCore .load_liquid_class ,
123
+ autospec = True ,
124
+ ) as patched_load_liquid_class ,
125
+ mock .patch .object (
126
+ InstrumentCore ,
127
+ "pick_up_tip" ,
128
+ side_effect = InstrumentCore .pick_up_tip ,
129
+ autospec = True ,
130
+ ) as patched_pick_up_tip ,
131
+ mock .patch .object (
132
+ InstrumentCore ,
133
+ "aspirate_liquid_class" ,
134
+ side_effect = InstrumentCore .aspirate_liquid_class ,
135
+ autospec = True ,
136
+ ) as patched_aspirate ,
137
+ mock .patch .object (
138
+ InstrumentCore ,
139
+ "dispense_liquid_class" ,
140
+ side_effect = InstrumentCore .dispense_liquid_class ,
141
+ autospec = True ,
142
+ ) as patched_dispense ,
143
+ mock .patch .object (
144
+ InstrumentCore ,
145
+ "drop_tip_in_disposal_location" ,
146
+ side_effect = InstrumentCore .drop_tip_in_disposal_location ,
147
+ autospec = True ,
148
+ ) as patched_drop_tip ,
149
+ ):
150
+ mock_manager = mock .Mock ()
151
+ mock_manager .attach_mock (patched_pick_up_tip , "pick_up_tip" )
152
+ mock_manager .attach_mock (patched_load_liquid_class , "load_liquid_class" )
153
+ mock_manager .attach_mock (patched_aspirate , "aspirate_liquid_class" )
154
+ mock_manager .attach_mock (patched_dispense , "dispense_liquid_class" )
155
+ mock_manager .attach_mock (patched_drop_tip , "drop_tip_in_disposal_location" )
156
+ pipette_50 .transfer_liquid (
157
+ liquid_class = water ,
158
+ volume = 40 ,
159
+ source = nest_plate .rows ()[0 ][:2 ],
160
+ dest = arma_plate .rows ()[0 ][:2 ],
161
+ new_tip = "always" ,
162
+ trash_location = trash ,
163
+ )
164
+ expected_calls = [
165
+ mock .call .load_liquid_class (
166
+ mock .ANY ,
167
+ name = "water" ,
168
+ transfer_properties = mock .ANY ,
169
+ tiprack_uri = "opentrons/opentrons_flex_96_tiprack_50ul/1" ,
170
+ ),
171
+ mock .call .pick_up_tip (
172
+ mock .ANY ,
173
+ location = mock .ANY ,
174
+ well_core = mock .ANY ,
175
+ presses = mock .ANY ,
176
+ increment = mock .ANY ,
177
+ ),
178
+ mock .call .aspirate_liquid_class (
179
+ mock .ANY ,
180
+ volume = 40 ,
181
+ source = mock .ANY ,
182
+ transfer_properties = mock .ANY ,
183
+ transfer_type = TransferType .ONE_TO_ONE ,
184
+ tip_contents = [LiquidAndAirGapPair (liquid = 0 , air_gap = 0 )],
185
+ ),
186
+ mock .call .dispense_liquid_class (
187
+ mock .ANY ,
188
+ volume = 40 ,
189
+ dest = mock .ANY ,
190
+ source = mock .ANY ,
191
+ transfer_properties = mock .ANY ,
192
+ transfer_type = TransferType .ONE_TO_ONE ,
193
+ tip_contents = [LiquidAndAirGapPair (liquid = 0 , air_gap = 0.1 )],
194
+ trash_location = mock .ANY ,
195
+ ),
196
+ mock .call .drop_tip_in_disposal_location (
197
+ mock .ANY ,
198
+ disposal_location = trash ,
199
+ home_after = False ,
200
+ alternate_tip_drop = True ,
201
+ ),
202
+ mock .call .pick_up_tip (
203
+ mock .ANY ,
204
+ location = mock .ANY ,
205
+ well_core = mock .ANY ,
206
+ presses = mock .ANY ,
207
+ increment = mock .ANY ,
208
+ ),
209
+ mock .call .aspirate_liquid_class (
210
+ mock .ANY ,
211
+ volume = 40 ,
212
+ source = mock .ANY ,
213
+ transfer_properties = mock .ANY ,
214
+ transfer_type = TransferType .ONE_TO_ONE ,
215
+ tip_contents = [LiquidAndAirGapPair (liquid = 0 , air_gap = 0 )],
216
+ ),
217
+ mock .call .dispense_liquid_class (
218
+ mock .ANY ,
219
+ volume = 40 ,
220
+ dest = mock .ANY ,
221
+ source = mock .ANY ,
222
+ transfer_properties = mock .ANY ,
223
+ transfer_type = TransferType .ONE_TO_ONE ,
224
+ tip_contents = [LiquidAndAirGapPair (liquid = 40 , air_gap = 0.1 )],
225
+ trash_location = mock .ANY ,
226
+ ),
227
+ mock .call .drop_tip_in_disposal_location (
228
+ mock .ANY ,
229
+ disposal_location = trash ,
230
+ home_after = False ,
231
+ alternate_tip_drop = True ,
232
+ ),
233
+ ]
234
+ assert len (mock_manager .mock_calls ) == 9
235
+ assert mock_manager .mock_calls == expected_calls
0 commit comments