Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

rough dynamic liquid tracking #17167

Draft
wants to merge 3 commits into
base: edge
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
16 changes: 16 additions & 0 deletions api/src/opentrons/hardware_control/motion_utilities.py
Original file line number Diff line number Diff line change
Expand Up @@ -192,6 +192,22 @@ def target_position_from_plunger(
return all_axes_pos


def target_positions_from_plunger_tracking(
mount: Union[Mount, OT3Mount],
plunger_delta: float,
z_delta: float,
current_position: Dict[Axis, float],
) -> "OrderedDict[Axis, float]":
"""Create a target position for machine axes including plungers.

The x/y axis remain constant but the plunger and Z move to create a tracking action.
"""
all_axes_pos = target_position_from_plunger(mount, plunger_delta, current_position)
z_ax = Axis.by_mount(mount)
all_axes_pos[z_ax] = current_position[z_ax] + z_delta
return all_axes_pos


def deck_point_from_machine_point(
machine_point: Point, attitude: AttitudeMatrix, offset: Point
) -> Point:
Expand Down
46 changes: 45 additions & 1 deletion api/src/opentrons/hardware_control/ot3api.py
Original file line number Diff line number Diff line change
Expand Up @@ -126,6 +126,7 @@
target_position_from_absolute,
target_position_from_relative,
target_position_from_plunger,
target_positions_from_plunger_tracking,
offset_for_mount,
deck_from_machine,
machine_from_deck,
Expand Down Expand Up @@ -2750,7 +2751,7 @@ async def liquid_probe( # noqa: C901
if not probe_settings:
probe_settings = deepcopy(self.config.liquid_sense)

# We need to significatly slow down the 96 channel liquid probe
# We need to significantly slow down the 96 channel liquid probe
if self.gantry_load == GantryLoad.HIGH_THROUGHPUT:
max_plunger_speed = self.config.motion_settings.max_speed_discontinuity[
GantryLoad.HIGH_THROUGHPUT
Expand Down Expand Up @@ -2964,6 +2965,49 @@ async def capacitive_sweep(

AMKey = TypeVar("AMKey")

async def aspirate_while_tracking(
self,
mount: Union[top_types.Mount, OT3Mount],
z_distance: float,
flow_rate: float,
volume: float,
) -> None:
"""
Aspirate a volume of liquid (in microliters/uL) using this pipette."""
realmount = OT3Mount.from_mount(mount)
aspirate_spec = self._pipette_handler.plan_check_aspirate(
realmount, volume, flow_rate
)
if not aspirate_spec:
return

target_pos = target_positions_from_plunger_tracking(
realmount,
aspirate_spec.plunger_distance,
z_distance,
self._current_position,
)

try:
await self._backend.set_active_current(
{aspirate_spec.axis: aspirate_spec.current}
)
async with self.restore_system_constrants():
await self.set_system_constraints_for_plunger_acceleration(
realmount, aspirate_spec.acceleration
)
await self._move(
target_pos,
speed=aspirate_spec.speed,
home_flagged_axes=False,
)
except Exception:
self._log.exception("Aspirate failed")
aspirate_spec.instr.set_current_volume(0)
raise
else:
aspirate_spec.instr.add_current_volume(aspirate_spec.volume)

@property
def attached_subsystems(self) -> Dict[SubSystem, SubSystemState]:
"""Get a view of the state of the currently-attached subsystems."""
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -215,3 +215,12 @@ async def liquid_probe(
max_z_dist : maximum depth to probe for liquid
"""
...

async def aspirate_while_tracking(
self,
mount: MountArgType,
z_distance: float,
flow_rate: float,
volume: float,
) -> None:
...
2 changes: 2 additions & 0 deletions api/src/opentrons/protocol_api/core/engine/instrument.py
Original file line number Diff line number Diff line change
Expand Up @@ -135,6 +135,7 @@ def aspirate(
flow_rate: float,
in_place: bool,
is_meniscus: Optional[bool] = None,
is_tracking: Optional[bool] = False,
) -> None:
"""Aspirate a given volume of liquid from the specified location.
Args:
Expand Down Expand Up @@ -192,6 +193,7 @@ def aspirate(
wellLocation=well_location,
volume=volume,
flowRate=flow_rate,
is_tracking=is_tracking if is_tracking else False,
)
)

Expand Down
1 change: 1 addition & 0 deletions api/src/opentrons/protocol_api/core/instrument.py
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,7 @@ def aspirate(
flow_rate: float,
in_place: bool,
is_meniscus: Optional[bool] = None,
is_tracking: Optional[bool] = False,
) -> None:
"""Aspirate a given volume of liquid from the specified location.
Args:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -85,6 +85,7 @@ def aspirate(
flow_rate: float,
in_place: bool,
is_meniscus: Optional[bool] = None,
is_tracking: Optional[bool] = None,
) -> None:
"""Aspirate a given volume of liquid from the specified location.
Args:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -96,6 +96,7 @@ def aspirate(
flow_rate: float,
in_place: bool,
is_meniscus: Optional[bool] = None,
is_tracking: Optional[bool] = None,
) -> None:
if self.get_current_volume() == 0:
# Make sure we're at the top of the labware and clear of any
Expand Down
131 changes: 131 additions & 0 deletions api/src/opentrons/protocol_api/instrument_context.py
Original file line number Diff line number Diff line change
Expand Up @@ -292,6 +292,137 @@ def aspirate(

return self

########

@requires_version(2, 0)
def aspirate_while_tracking(
self,
volume: Optional[float] = None,
location: Optional[Union[types.Location, labware.Well]] = None,
rate: float = 1.0,
) -> InstrumentContext:
"""
Draw liquid into a pipette tip.

See :ref:`new-aspirate` for more details and examples.

:param volume: The volume to aspirate, measured in µL. If unspecified,
defaults to the maximum volume for the pipette and its currently
attached tip.

If ``aspirate`` is called with a volume of precisely 0, its behavior
depends on the API level of the protocol. On API levels below 2.16,
it will behave the same as a volume of ``None``/unspecified: aspirate
until the pipette is full. On API levels at or above 2.16, no liquid
will be aspirated.
:type volume: int or float
:param location: Tells the robot where to aspirate from. The location can be
a :py:class:`.Well` or a :py:class:`.Location`.

- If the location is a ``Well``, the robot will aspirate at
or above the bottom center of the well. The distance (in mm)
from the well bottom is specified by
:py:obj:`well_bottom_clearance.aspirate
<well_bottom_clearance>`.

- If the location is a ``Location`` (e.g., the result of
:py:meth:`.Well.top` or :py:meth:`.Well.bottom`), the robot
will aspirate from that specified position.

- If the ``location`` is unspecified, the robot will
aspirate from its current position.
:param rate: A multiplier for the default flow rate of the pipette. Calculated
as ``rate`` multiplied by :py:attr:`flow_rate.aspirate
<flow_rate>`. If not specified, defaults to 1.0. See
:ref:`new-plunger-flow-rates`.
:type rate: float
:returns: This instance.

.. note::

If ``aspirate`` is called with a single, unnamed argument, it will treat
that argument as ``volume``. If you want to call ``aspirate`` with only
``location``, specify it as a keyword argument:
``pipette.aspirate(location=plate['A1'])``

"""
_log.debug(
"aspirate {} from {} at {}".format(
volume, location if location else "current position", rate
)
)

move_to_location: types.Location
well: Optional[labware.Well] = None
is_meniscus: Optional[bool] = None
last_location = self._get_last_location_by_api_version()
try:
target = validation.validate_location(
location=location, last_location=last_location
)
except validation.NoLocationError as e:
raise RuntimeError(
"If aspirate is called without an explicit location, another"
" method that moves to a location (such as move_to or "
"dispense) must previously have been called so the robot "
"knows where it is."
) from e

if isinstance(target, (TrashBin, WasteChute)):
raise ValueError(
"Trash Bin and Waste Chute are not acceptable location parameters for Aspirate commands."
)
move_to_location, well, is_meniscus = self._handle_aspirate_target(
target=target
)
if self.api_version >= APIVersion(2, 11):
instrument.validate_takes_liquid(
location=move_to_location,
reject_module=self.api_version >= APIVersion(2, 13),
reject_adapter=self.api_version >= APIVersion(2, 15),
)

if self.api_version >= APIVersion(2, 16):
c_vol = self._core.get_available_volume() if volume is None else volume
else:
c_vol = self._core.get_available_volume() if not volume else volume
flow_rate = self._core.get_aspirate_flow_rate(rate)

if (
self.api_version >= APIVersion(2, 20)
and well is not None
and self.liquid_presence_detection
and self._core.nozzle_configuration_valid_for_lld()
and self._core.get_current_volume() == 0
):
self._raise_if_pressure_not_supported_by_pipette()
self.require_liquid_presence(well=well)

with publisher.publish_context(
broker=self.broker,
command=cmds.aspirate(
instrument=self,
volume=c_vol,
location=move_to_location,
flow_rate=flow_rate,
rate=rate,
),
):
self._core.aspirate(
location=move_to_location,
well_core=well._core if well is not None else None,
volume=c_vol,
rate=rate,
flow_rate=flow_rate,
in_place=target.in_place,
is_meniscus=is_meniscus,
is_tracking=True,
)

return self

########

@requires_version(2, 0)
def dispense( # noqa: C901
self,
Expand Down
58 changes: 42 additions & 16 deletions api/src/opentrons/protocol_engine/commands/aspirate.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,9 @@
FlowRateMixin,
BaseLiquidHandlingResult,
aspirate_in_place,
aspirate_while_tracking,
prepare_for_aspirate,
IsTrackingMixin,
)
from .movement_common import (
LiquidHandlingWellLocationMixin,
Expand Down Expand Up @@ -47,7 +49,11 @@


class AspirateParams(
PipetteIdMixin, AspirateVolumeMixin, FlowRateMixin, LiquidHandlingWellLocationMixin
PipetteIdMixin,
AspirateVolumeMixin,
FlowRateMixin,
LiquidHandlingWellLocationMixin,
IsTrackingMixin,
):
"""Parameters required to aspirate from a specific well."""

Expand Down Expand Up @@ -158,28 +164,48 @@ async def execute(self, params: AspirateParams) -> _ExecuteReturn:
well_location=well_location,
current_well=current_well,
operation_volume=-params.volume,
is_tracking=params.is_tracking,
)
state_update.append(move_result.state_update)
if isinstance(move_result, DefinedErrorData):
return DefinedErrorData(
public=move_result.public, state_update=state_update
)

aspirate_result = await aspirate_in_place(
pipette_id=pipette_id,
volume=params.volume,
flow_rate=params.flowRate,
location_if_error={
"retryLocation": (
move_result.public.position.x,
move_result.public.position.y,
move_result.public.position.z,
)
},
command_note_adder=self._command_note_adder,
pipetting=self._pipetting,
model_utils=self._model_utils,
)
if params.is_tracking:
aspirate_result = await aspirate_while_tracking(
pipette_id=pipette_id,
labware_id=labware_id,
well_name=well_name,
volume=params.volume,
flow_rate=params.flowRate,
location_if_error={
"retryLocation": (
move_result.public.position.x,
move_result.public.position.y,
move_result.public.position.z,
)
},
command_note_adder=self._command_note_adder,
pipetting=self._pipetting,
model_utils=self._model_utils,
)
else:
aspirate_result = await aspirate_in_place(
pipette_id=pipette_id,
volume=params.volume,
flow_rate=params.flowRate,
location_if_error={
"retryLocation": (
move_result.public.position.x,
move_result.public.position.y,
move_result.public.position.z,
)
},
command_note_adder=self._command_note_adder,
pipetting=self._pipetting,
model_utils=self._model_utils,
)
state_update.append(aspirate_result.state_update)
if isinstance(aspirate_result, DefinedErrorData):
state_update.set_liquid_operated(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -152,6 +152,7 @@ async def move_to_well(
minimum_z_height: Optional[float] = None,
speed: Optional[float] = None,
operation_volume: Optional[float] = None,
is_tracking: Optional[bool] = False,
) -> MoveToWellOperationReturn:
"""Execute a move to well microoperation."""
try:
Expand All @@ -165,6 +166,7 @@ async def move_to_well(
minimum_z_height=minimum_z_height,
speed=speed,
operation_volume=operation_volume,
is_tracking=is_tracking,
)
except StallOrCollisionDetectedError as e:
return DefinedErrorData(
Expand Down
Loading
Loading