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

feat(api, robot-server): Plate Reader CSV output functionality #16495

Merged
merged 12 commits into from
Oct 19, 2024
Merged
1 change: 1 addition & 0 deletions api/src/opentrons/cli/analyze.py
Original file line number Diff line number Diff line change
Expand Up @@ -332,6 +332,7 @@ async def _do_analyze(
liquids=[],
wells=[],
hasEverEnteredErrorRecovery=False,
files=[],
),
parameters=[],
)
Expand Down
1 change: 1 addition & 0 deletions api/src/opentrons/execute.py
Original file line number Diff line number Diff line change
Expand Up @@ -546,6 +546,7 @@ def _create_live_context_pe(
hardware_api=hardware_api_wrapped,
config=_get_protocol_engine_config(),
deck_configuration=entrypoint_util.get_deck_configuration(),
file_provider=None,
error_recovery_policy=error_recovery_policy.never_recover,
drop_tips_after_run=False,
post_run_hardware_state=PostRunHardwareState.STAY_ENGAGED_IN_PLACE,
Expand Down
27 changes: 23 additions & 4 deletions api/src/opentrons/protocol_api/core/engine/module_core.py
Original file line number Diff line number Diff line change
Expand Up @@ -586,11 +586,20 @@ def initialize(
)
self._initialized_value = wavelengths

def read(self) -> Optional[Dict[int, Dict[str, float]]]:
"""Initiate a read on the Absorbance Reader, and return the results. During Analysis, this will return None."""
def read(self, filename: Optional[str]) -> Dict[int, Dict[str, float]]:
"""Initiate a read on the Absorbance Reader, and return the results. During Analysis, this will return a measurement of zero for all wells."""
wavelengths = self._engine_client.state.modules.get_absorbance_reader_substate(
self.module_id
).configured_wavelengths
if wavelengths is None:
raise CannotPerformModuleAction(
"Cannot perform Read action on Absorbance Reader without calling `.initialize(...)` first."
)
if self._initialized_value:
self._engine_client.execute_command(
cmd.absorbance_reader.ReadAbsorbanceParams(moduleId=self.module_id)
cmd.absorbance_reader.ReadAbsorbanceParams(
moduleId=self.module_id, fileName=filename
)
)
if not self._engine_client.state.config.use_virtual_modules:
read_result = (
Expand All @@ -603,7 +612,17 @@ def read(self) -> Optional[Dict[int, Dict[str, float]]]:
raise CannotPerformModuleAction(
"Absorbance Reader failed to return expected read result."
)
return None

# When using virtual modules, return all zeroes
virtual_asbsorbance_result: Dict[int, Dict[str, float]] = {}
for wavelength in wavelengths:
converted_values = (
self._engine_client.state.modules.convert_absorbance_reader_data_points(
data=[0] * 96
)
)
virtual_asbsorbance_result[wavelength] = converted_values
return virtual_asbsorbance_result

def close_lid(
self,
Expand Down
2 changes: 1 addition & 1 deletion api/src/opentrons/protocol_api/core/module.py
Original file line number Diff line number Diff line change
Expand Up @@ -365,7 +365,7 @@ def initialize(
"""Initialize the Absorbance Reader by taking zero reading."""

@abstractmethod
def read(self) -> Optional[Dict[int, Dict[str, float]]]:
def read(self, filename: Optional[str]) -> Dict[int, Dict[str, float]]:
"""Get an absorbance reading from the Absorbance Reader."""

@abstractmethod
Expand Down
9 changes: 6 additions & 3 deletions api/src/opentrons/protocol_api/module_contexts.py
Original file line number Diff line number Diff line change
Expand Up @@ -1035,6 +1035,9 @@ def initialize(
)

@requires_version(2, 21)
def read(self) -> Optional[Dict[int, Dict[str, float]]]:
"""Initiate read on the Absorbance Reader. Returns a dictionary of wavelengths to dictionary of values ordered by well name."""
return self._core.read()
def read(self, filename: Optional[str]) -> Dict[int, Dict[str, float]]:
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would consider naming the filename parameter something like save_read_results_to_filename.

Right now, module.read(filename="foo") really makes it look like you're reading from an existing file called "foo".

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

would something shorter like export_filename or export_to_filename work (preference to the first for briefness, but I think clarity is important).

"""Initiate read on the Absorbance Reader. Returns a dictionary of wavelengths to dictionary of values ordered by well name.

:param filename: Optional, if a filename is provided a CSV file will be saved as a result of the read action containing measurement data.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does this mean that if filename isn't specified…

  1. The file will get a default name like Plate Reader output [timestamp].csv
  2. No file will be written 🚮

My understanding was that we'd always write a file.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No file will be written. There is a limit (currently) of 40 files written during a protocol run. The user can do unlimited reads so long as they are not writing a file. They can do 40 reads where they save a file.

"""
return self._core.read(filename=filename)
Original file line number Diff line number Diff line change
@@ -1,14 +1,22 @@
"""Command models to read absorbance."""
from __future__ import annotations
from typing import Optional, Dict, TYPE_CHECKING
from datetime import datetime
from typing import Optional, Dict, TYPE_CHECKING, List
from typing_extensions import Literal, Type

from pydantic import BaseModel, Field

from ..command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData
from ...errors import CannotPerformModuleAction
from ...errors import CannotPerformModuleAction, StorageLimitReachedError
from ...errors.error_occurrence import ErrorOccurrence

from ...resources.file_provider import (
PlateReaderData,
ReadData,
MAXIMUM_CSV_FILE_LIMIT,
)
from ...resources import FileProvider

if TYPE_CHECKING:
from opentrons.protocol_engine.state.state import StateView
from opentrons.protocol_engine.execution import EquipmentHandler
Expand All @@ -21,6 +29,10 @@ class ReadAbsorbanceParams(BaseModel):
"""Input parameters for an absorbance reading."""

moduleId: str = Field(..., description="Unique ID of the Absorbance Reader.")
fileName: Optional[str] = Field(
None,
description="Optional file name to use when storing the results of a measurement.",
)


class ReadAbsorbanceResult(BaseModel):
Expand All @@ -29,6 +41,10 @@ class ReadAbsorbanceResult(BaseModel):
data: Optional[Dict[int, Dict[str, float]]] = Field(
..., description="Absorbance data points per wavelength."
)
fileIds: Optional[List[str]] = Field(
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nice

...,
description="List of file IDs for files output as a result of a Read action.",
SyntaxColoring marked this conversation as resolved.
Show resolved Hide resolved
)


class ReadAbsorbanceImpl(
Expand All @@ -40,18 +56,21 @@ def __init__(
self,
state_view: StateView,
equipment: EquipmentHandler,
file_provider: FileProvider,
**unused_dependencies: object,
) -> None:
self._state_view = state_view
self._equipment = equipment
self._file_provider = file_provider

async def execute(
async def execute( # noqa: C901
self, params: ReadAbsorbanceParams
) -> SuccessData[ReadAbsorbanceResult, None]:
"""Initiate an absorbance measurement."""
abs_reader_substate = self._state_view.modules.get_absorbance_reader_substate(
module_id=params.moduleId
)

# Allow propagation of ModuleNotAttachedError.
abs_reader = self._equipment.get_module_hardware_api(
abs_reader_substate.module_id
Expand All @@ -62,10 +81,29 @@ async def execute(
"Cannot perform Read action on Absorbance Reader without calling `.initialize(...)` first."
)

# TODO: we need to return a file ID and increase the file count even when a moduel is not attached
if (
params.fileName is not None
and abs_reader_substate.configured_wavelengths is not None
):
# Validate that the amount of files we are about to generate does not put us higher than the limit
if (
self._state_view.files.get_filecount()
+ len(abs_reader_substate.configured_wavelengths)
> MAXIMUM_CSV_FILE_LIMIT
):
raise StorageLimitReachedError(
message=f"Attempt to write file {params.fileName} exceeds file creation limit of {MAXIMUM_CSV_FILE_LIMIT} files."
)

asbsorbance_result: Dict[int, Dict[str, float]] = {}
transform_results = []
# Handle the measurement and begin building data for return
if abs_reader is not None:
start_time = datetime.now()
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You probably want to use ModelUtils.get_timestamp() instead of a raw datetime.now(), for two reasons:

  1. Dependency injection (testability).
  2. Consistent timezone.

results = await abs_reader.start_measure()
finish_time = datetime.now()
if abs_reader._measurement_config is not None:
asbsorbance_result: Dict[int, Dict[str, float]] = {}
sample_wavelengths = abs_reader._measurement_config.sample_wavelengths
for wavelength, result in zip(sample_wavelengths, results):
converted_values = (
Expand All @@ -74,13 +112,67 @@ async def execute(
)
)
asbsorbance_result[wavelength] = converted_values
transform_results.append(
ReadData.construct(wavelength=wavelength, data=converted_values)
)
# Handle the virtual module case for data creation (all zeroes)
elif self._state_view.config.use_virtual_modules:
start_time = finish_time = datetime.now()
if abs_reader_substate.configured_wavelengths is not None:
for wavelength in abs_reader_substate.configured_wavelengths:
converted_values = (
self._state_view.modules.convert_absorbance_reader_data_points(
data=[0] * 96
)
)
asbsorbance_result[wavelength] = converted_values
transform_results.append(
ReadData.construct(wavelength=wavelength, data=converted_values)
)
else:
raise CannotPerformModuleAction(
"Plate Reader data cannot be requested with a module that has not been initialized."
)

# TODO (cb, 10-17-2024): FILE PROVIDER - Some day we may want to break the file provider behavior into a seperate API function.
# When this happens, we probably will to have the change the command results handler we utilize to track file IDs in engine.
# Today, the action handler for the FileStore looks for a ReadAbsorbanceResult command action, this will need to be delinked.

# Begin interfacing with the file provider if the user provided a filename
file_ids = []
if params.fileName is not None:
# Create the Plate Reader Transform
plate_read_result = PlateReaderData.construct(
read_results=transform_results,
reference_wavelength=abs_reader_substate.reference_wavelength,
start_time=start_time,
finish_time=finish_time,
serial_number=abs_reader.serial_number
if (abs_reader is not None and abs_reader.serial_number is not None)
else "VIRTUAL_SERIAL",
)

if isinstance(plate_read_result, PlateReaderData):
# Write a CSV file for each of the measurements taken
for measurement in plate_read_result.read_results:
file_id = await self._file_provider.write_csv(
write_data=plate_read_result.build_generic_csv(
filename=params.fileName,
measurement=measurement,
)
)
file_ids.append(file_id)

# Return success data to api
return SuccessData(
public=ReadAbsorbanceResult(data=asbsorbance_result),
public=ReadAbsorbanceResult(
data=asbsorbance_result, fileIds=file_ids
),
private=None,
)

return SuccessData(
public=ReadAbsorbanceResult(data=None),
public=ReadAbsorbanceResult(data=asbsorbance_result, fileIds=file_ids),
private=None,
)

Expand Down
1 change: 1 addition & 0 deletions api/src/opentrons/protocol_engine/commands/command.py
Original file line number Diff line number Diff line change
Expand Up @@ -274,6 +274,7 @@ def __init__(
state_view: StateView,
hardware_api: HardwareControlAPI,
equipment: execution.EquipmentHandler,
file_provider: execution.FileProvider,
movement: execution.MovementHandler,
gantry_mover: execution.GantryMover,
labware_movement: execution.LabwareMovementHandler,
Expand Down
10 changes: 9 additions & 1 deletion api/src/opentrons/protocol_engine/create_protocol_engine.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@
from opentrons_shared_data.robot import load as load_robot

from .protocol_engine import ProtocolEngine
from .resources import DeckDataProvider, ModuleDataProvider
from .resources import DeckDataProvider, ModuleDataProvider, FileProvider
from .state.config import Config
from .state.state import StateStore
from .types import PostRunHardwareState, DeckConfigurationType
Expand All @@ -26,6 +26,7 @@ async def create_protocol_engine(
error_recovery_policy: ErrorRecoveryPolicy,
load_fixed_trash: bool = False,
deck_configuration: typing.Optional[DeckConfigurationType] = None,
file_provider: typing.Optional[FileProvider] = None,
notify_publishers: typing.Optional[typing.Callable[[], None]] = None,
) -> ProtocolEngine:
"""Create a ProtocolEngine instance.
Expand All @@ -37,6 +38,7 @@ async def create_protocol_engine(
See documentation on `ErrorRecoveryPolicy`.
load_fixed_trash: Automatically load fixed trash labware in engine.
deck_configuration: The initial deck configuration the engine will be instantiated with.
file_provider: Provides access to robot server file writing procedures for protocol output.
notify_publishers: Notifies robot server publishers of internal state change.
"""
deck_data = DeckDataProvider(config.deck_type)
Expand All @@ -47,6 +49,7 @@ async def create_protocol_engine(

module_calibration_offsets = ModuleDataProvider.load_module_calibrations()
robot_definition = load_robot(config.robot_type)

state_store = StateStore(
config=config,
deck_definition=deck_definition,
Expand All @@ -62,6 +65,7 @@ async def create_protocol_engine(
return ProtocolEngine(
state_store=state_store,
hardware_api=hardware_api,
file_provider=file_provider,
)


Expand All @@ -70,6 +74,7 @@ def create_protocol_engine_in_thread(
hardware_api: HardwareControlAPI,
config: Config,
deck_configuration: typing.Optional[DeckConfigurationType],
file_provider: typing.Optional[FileProvider],
error_recovery_policy: ErrorRecoveryPolicy,
drop_tips_after_run: bool,
post_run_hardware_state: PostRunHardwareState,
Expand Down Expand Up @@ -97,6 +102,7 @@ def create_protocol_engine_in_thread(
with async_context_manager_in_thread(
_protocol_engine(
hardware_api,
file_provider,
config,
deck_configuration,
error_recovery_policy,
Expand All @@ -114,6 +120,7 @@ def create_protocol_engine_in_thread(
@contextlib.asynccontextmanager
async def _protocol_engine(
hardware_api: HardwareControlAPI,
file_provider: typing.Optional[FileProvider],
config: Config,
deck_configuration: typing.Optional[DeckConfigurationType],
error_recovery_policy: ErrorRecoveryPolicy,
Expand All @@ -123,6 +130,7 @@ async def _protocol_engine(
) -> typing.AsyncGenerator[ProtocolEngine, None]:
protocol_engine = await create_protocol_engine(
hardware_api=hardware_api,
file_provider=file_provider,
config=config,
error_recovery_policy=error_recovery_policy,
load_fixed_trash=load_fixed_trash,
Expand Down
3 changes: 2 additions & 1 deletion api/src/opentrons/protocol_engine/engine_support.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,8 @@


def create_run_orchestrator(
hardware_api: HardwareControlAPI, protocol_engine: ProtocolEngine
hardware_api: HardwareControlAPI,
protocol_engine: ProtocolEngine,
) -> RunOrchestrator:
"""Create a RunOrchestrator instance."""
return RunOrchestrator(
Expand Down
2 changes: 2 additions & 0 deletions api/src/opentrons/protocol_engine/errors/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,7 @@
InvalidLiquidHeightFound,
LiquidHeightUnknownError,
InvalidWellDefinitionError,
StorageLimitReachedError,
)

from .error_occurrence import ErrorOccurrence, ProtocolCommandFailedError
Expand Down Expand Up @@ -152,4 +153,5 @@
"InvalidLiquidHeightFound",
"LiquidHeightUnknownError",
"InvalidWellDefinitionError",
"StorageLimitReachedError",
]
13 changes: 13 additions & 0 deletions api/src/opentrons/protocol_engine/errors/exceptions.py
Original file line number Diff line number Diff line change
Expand Up @@ -1082,3 +1082,16 @@ def __init__(
) -> None:
"""Build an InvalidWellDefinitionError."""
super().__init__(ErrorCodes.GENERAL_ERROR, message, details, wrapping)


class StorageLimitReachedError(ProtocolEngineError):
"""Raised to indicate that a file cannot be created due to storage limitations."""

def __init__(
self,
message: Optional[str] = None,
detail: Optional[Dict[str, str]] = None,
wrapping: Optional[Sequence[EnumeratedError]] = None,
) -> None:
"""Build an StorageLimitReached."""
super().__init__(ErrorCodes.GENERAL_ERROR, message, detail, wrapping)
2 changes: 2 additions & 0 deletions api/src/opentrons/protocol_engine/execution/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@
from .hardware_stopper import HardwareStopper
from .door_watcher import DoorWatcher
from .status_bar import StatusBarHandler
from ..resources.file_provider import FileProvider

# .thermocycler_movement_flagger omitted from package's public interface.

Expand All @@ -45,4 +46,5 @@
"DoorWatcher",
"RailLightsHandler",
"StatusBarHandler",
"FileProvider",
]
Loading
Loading