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): add InstrumentContext.load_name and update LiquidClass.get_for() #17289

Open
wants to merge 1 commit into
base: edge
Choose a base branch
from

Conversation

sanni-t
Copy link
Member

@sanni-t sanni-t commented Jan 16, 2025

Closes AUTH-1295, AUTH-1299

Overview

  • adds InstrumentContext.load_name to fetch the python API load name of a pipette
  • updates LiquidClass.get_for() to accept the a loaded instrument object and loaded tiprack object for pipette and tiprack args respectively.

Test Plan and Hands on Testing

Added unit and integration tests

Review requests

  • Note that the LiquidClass.get_for() method will still accept the pipette name and tiprack URI as arguments. This is to allow fetching liquid class properties without needing to load pipettes and tipracks. Let me know if this is not a good idea
  • Any naming alternatives to InstrumentContext.load_name? I want to avoid any confusion with the existing name property and this new property

Risk assessment

None

@sanni-t sanni-t requested a review from a team as a code owner January 16, 2025 17:43
@ddcc4
Copy link
Contributor

ddcc4 commented Jan 17, 2025

Any naming alternatives to InstrumentContext.load_name? I want to avoid any confusion with the existing name property and this new property

Don't we already consistently call it load_name throughout the docs? What's the concern about calling it load_name here then?

"""Get liquid class transfer properties for the specified pipette and tip."""
from . import InstrumentContext, Labware
Copy link
Contributor

Choose a reason for hiding this comment

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

Why can't we just import this at the top of the file (and get rid of the if TYPE_CHECKING bit)?

You are clearly using the symbols InstrumentContext and Labware in your implementation (versus only importing it for the typechecker), so why not treat it as a normal import?

Copy link
Contributor

Choose a reason for hiding this comment

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

Adding to this, I can see limited use of getting liquid class properties without having to provide the names, but I'd almost rather that be it's own function which would allow us to remove this internal import. That or we could always flip around the isinstance to check for str instead of InstrumentContext or Labware

Copy link
Member Author

@sanni-t sanni-t Jan 17, 2025

Choose a reason for hiding this comment

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

Why can't we just import this at the top of the file (and get rid of the if TYPE_CHECKING bit)?

Because it results in a circular import for InstrumentContext. InstrumentContext already imports from _liquid.

I can see limited use of getting liquid class properties without having to provide the names, but I'd almost rather that be it's own function

Not sure I understand. Can you elaborate?

That or we could always flip around the isinstance to check for str instead of InstrumentContext or Labware

If we want to verify that the args are not of some third, unsupported type then we will have to check if isinstance of InstrumentContext and Labware. Sure we can maybe check that if it's not a string and if there's an error doing pipette.load_name or tiprack.uri, the args are wrong. But it's not as foolproof as doing isinstance checks.

Copy link
Contributor

@SyntaxColoring SyntaxColoring left a comment

Choose a reason for hiding this comment

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

TY! Looks good except for a possible GEN1/GEN2 cross-compatibility thing.

@@ -64,18 +67,42 @@ def name(self) -> str:
def display_name(self) -> str:
return self._display_name

def get_for(self, pipette: str, tiprack: str) -> TransferProperties:
def get_for(
self, pipette: Union[str, InstrumentContext], tiprack: Union[str, Labware]
Copy link
Contributor

Choose a reason for hiding this comment

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

Nit: "tip rack" is two words, so can we name the argument tip_rack instead of tiprack?

Comment on lines +416 to +422
def get_load_name(self) -> str:
"""Get the pipette's requested API load name.

For OT-2 pipettes, this is the same as pipette name.
"""
return self.get_hardware_state()["name"]

Copy link
Contributor

Choose a reason for hiding this comment

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

For OT-2 pipettes, this is the same as pipette name.

Not always, I don't think. What about GEN1/GEN2 cross-compatibility? If a protocol requests p300_single but gets a p300_single_gen2, I'd expect load_name to return p300_single. I think this get_hardware_state()-based implementation will return p300_single_gen2.

I don't have a robot in front of me to test this, so feel free to dismiss this change request if I'm wrong about this.

Copy link
Member Author

Choose a reason for hiding this comment

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

This is what I was missing! The description of InstrumentCore.get_pipette_name() (which is what InstrumentContext.name returns) says

Will match the load name of the actually loaded pipette, which may differ from the requested load name.

Now I know what it means.
I actually want to return the name of the loaded pipette p300_single_gen2 and not the requested pipette p300_single.

I feel like fixing InstrumentContext.name might be the more appropriate thing here. We can gate the change behind API version. Otherwise InstrumentContext.name would just be something that returns the same thing as InstrumentContext.load_name for OT-2 pipettes and something unusable for the API users when using a Flex.

Copy link
Contributor

Choose a reason for hiding this comment

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

👍 Makes sense.

Copy link
Contributor

Choose a reason for hiding this comment

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

Sorry, now I'm confused, but I wasn't following this too closely.

When public users use liquid classes, they'll call get_for(pipette_name, tiprack_name), right?

In this example here, is the pipette_name going to be p300_single or p300_single_gen2?

(If your answer is p300_single_gen2, won't that be confusing to the user?)

Comment on lines +192 to +200
@abstractmethod
def get_load_name(self) -> str:
"""Get the pipette's requested API load name.

This is the load name that is specified in the `ProtocolContext.load_instrument()`
method. This name might differ from the engine-specific pipette name.
"""
...

Copy link
Contributor

Choose a reason for hiding this comment

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

Your call, but I wonder if this is easier to implement directly in InstrumentContext instead of in the InstrumentCores.

MAX_SUPPORTED_VERSION = APIVersion(2, 22)
MAX_SUPPORTED_VERSION = APIVersion(2, 23)
Copy link
Contributor

Choose a reason for hiding this comment

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

Just making sure you mean to have this in this PR?

Copy link
Member Author

Choose a reason for hiding this comment

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

It's intentional. But I did forget to call this out in the PR description and internal notes. Will do that

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants