Atomic Liquid Handling

Tip Handling

When we handle liquids with a pipette, we are constantly exchanging old, used tips for new ones to prevent cross-contamination between our wells. To help with this constant need, we describe in this section a few methods for getting new tips, and removing tips from a pipette.


This section demonstrates the options available for controlling tips

'''
Examples in this section expect the following
'''
from opentrons import labware, instruments, robot

tiprack = labware.load("opentrons_96_tiprack_300ul", "2")

pipette = instruments.P300_Single(mount="left")

Pick Up Tip

Before any liquid handling can be done, your pipette must have a tip on it. The command pick_up_tip() will move the pipette over to the specified tip, the press down into it to create a vacuum seal. The below example picks up the tip at location "A1".

pipette.pick_up_tip(tiprack.wells("A1"))

Drop Tip

Once finished with a tip, the pipette will autonomously remove the tip when we call drop_tip(). We can specify where to drop the tip by passing in a location. The below example drops the tip back at its originating location on the tip rack. If no location is specified, it will go to the fixed trash location on the deck.

pipette.drop_tip(tiprack.wells("A1"))

Instead of returning a tip to the tip rack, we can also drop it in an alternative trash container besides the fixed trash on the deck.

trash = labware.load("trash-box", "1")
pipette.pick_up_tip(tiprack.wells("A2"))
pipette.drop_tip(trash)

Return Tip

When we need to return the tip to its originating location on the tip rack, we can simply call return_tip(). The example below will automatically return the tip to "A3" on the tip rack.

pipette.pick_up_tip(tiprack.wells("A3"))
pipette.return_tip()

Tips Iterating

Automatically iterate through tips and drop tip in trash by attaching containers to a pipette. If no location is specified, the pipette will move to the next available tip by iterating through the tiprack that is associated with it.

'''
Examples in this section expect the following
'''
from opentrons import labware, instruments, robot

trash = labware.load("trash-box", "1")
tip_rack_1 = containers.load("opentrons_96_tiprack_300ul", "2")
tip_rack_2 = containers.load("opentrons_96_tiprack_300ul", "3")

Attach Tip Rack to Pipette

Tip racks and trash containers can be “attached” to a pipette when the pipette is created. This give the pipette the ability to automatically iterate through tips, and to automatically send the tip to the trash container.

Trash containers can be attached with the option trash_container=TRASH_CONTAINER.

Multiple tip racks are can be attached with the option tip_racks=[RACK_1, RACK_2, etc... ].

pipette = instruments.P300_Single(mount="left",
                                  tip_racks=[tip_rack_1, tip_rack_2],
                                  trash_container=trash)

Note

The tip_racks= option expects us to give it a Python list, containing each tip rack we want to attach. If we are only attaching one tip rack, then the list will have a length of one, like the following:

tip_racks=[tiprack]

Iterating Through Tips

Now that we have two tip racks attached to the pipette, we can automatically step through each tip whenever we call pick_up_tip(). We then have the option to either return_tip() to the tip rack, or we can drop_tip() to remove the tip in the attached trash container.

pipette.pick_up_tip()  # picks up tip_rack_1:A1
pipette.return_tip()
pipette.pick_up_tip()  # picks up tip_rack_1:A2
pipette.drop_tip()     # automatically drops in trash

# use loop to pick up tips tip_rack_1:A3 through tip_rack_2:H12
tips_left = 94 + 96 # add up the number of tips leftover in both tipracks
for _ in range(tips_left):
    pipette.pick_up_tip()
    pipette.return_tip()

If we try to pick_up_tip() again when all the tips have been used, the Opentrons API will show you an error.

Note

If you run the cell above, and then uncomment and run the cell below, you will get an error because the pipette is out of tips.

# this will raise an exception if run after the previous code block
# pipette.pick_up_tip()

Reseting Tip Tracking

If you plan to change out tipracks during the protocol run, you must reset tip tracking to prevent any errors. This is done through pipette.reset() which resets the tipracks and sets the current volume back to 0 ul.

pipette.reset()

Select Starting Tip

Calls to pick_up_tip() will by default start at the attached tip rack’s "A1" location in order of tipracks listed. If you however want to start automatic tip iterating at a different tip, you can use start_at_tip().

pipette.start_at_tip(tip_rack_1.well("C3"))
pipette.pick_up_tip()  # pick up C3 from "tip_rack_1"
pipette.return_tip()

Get Current Tip

Get the source location of the pipette’s current tip by calling current_tip(). If the tip was from the "A1" position on our tip rack, current_tip() will return that position.

print(pipette.current_tip())  # is holding no tip

pipette.pick_up_tip()
print(pipette.current_tip())  # is holding the next available tip

pipette.return_tip()
print(pipette.current_tip())  # is holding no tip

will print out…


Liquid Control

This is the fun section, where we get to move things around and pipette! This section describes the Pipette object’s many liquid-handling commands, as well as how to move the robot. Please note that the default now for pipette aspirate and dispense location is a 1mm offset from the bottom of the well now.


from opentrons import labware, instruments, robot

'''
Examples in this section expect the following:
'''
plate = labware.load("96-flat", "1")
pipette = instruments.P300_Single(mount="left")
pipette.pick_up_tip()

Aspirate

To aspirate is to pull liquid up into the pipette’s tip. When calling aspirate on a pipette, we can specify how many micoliters, and at which location, to draw liquid from:

pipette.aspirate(50, plate.wells("A1"))  # aspirate 50uL from plate:A1

Now our pipette’s tip is holding 50uL.

We can also simply specify how many microliters to aspirate, and not mention a location. The pipette in this circumstance will aspirate from it’s current location (which we previously set as plate.wells("A1")).

pipette.aspirate(50)                     # aspirate 50uL from current position

Now our pipette’s tip is holding 100uL.

We can also specify only the location to aspirate from. If we do not tell the pipette how many micoliters to aspirate, it will by default fill up the remaining volume in it’s tip. In this example, since we already have 100uL in the tip, the pipette will aspirate another 200uL

pipette.aspirate(plate.wells("A2"))      # aspirate until pipette fills from plate:A2

Dispense

To dispense is to push out liquid from the pipette’s tip. It’s usage in the Opentrons API is nearly identical to aspirate(), in that you can specify microliters and location, only microliters, or only a location:

pipette.dispense(50, plate.wells("B1")) # dispense 50uL to plate:B1
pipette.dispense(50)                    # dispense 50uL to current position
pipette.dispense(plate.wells("B2"))     # dispense until pipette empties to plate:B2

That final dispense without specifying a micoliter amount will dispense all remaining liquids in the tip to plate.wells("B2"), and now our pipette is empty.

Blow Out

To blow out is to push an extra amount of air through the pipette’s tip, so as to make sure that any remaining droplets are expelled.

When calling blow_out() on a pipette, we have the option to specify a location to blow out the remaining liquid. If no location is specified, the pipette will blow out from it’s current position.

pipette.blow_out()                  # blow out in current location
pipette.blow_out(plate.wells("B3")) # blow out in current plate:B3

Touch Tip

To touch tip is to move the pipette’s currently attached tip to four opposite edges of a well, for the purpose of knocking off any droplets that might be hanging from the tip.

When calling touch_tip() on a pipette, we have the option to specify a location where the tip will touch the inner walls. If no location is specified, the pipette will touch tip inside it’s current location.

pipette.touch_tip()                  # touch tip within current location
pipette.touch_tip(v_offset=-2)       # touch tip 2mm below the top of the current location
pipette.touch_tip(plate.wells("B1")) # touch tip within plate:B1

Mix

Mixing is simply performing a series of aspirate() and dispense() commands in a row on a single location. However, instead of having to write those commands out every time, the Opentrons API allows you to simply say mix().

The mix command takes three arguments: mix(repetitions, volume, location)

pipette.mix(4, 100, plate.wells("A2"))   # mix 4 times, 100uL, in plate:A2
pipette.mix(3, 50)                       # mix 3 times, 50uL, in current location
pipette.mix(2)                           # mix 2 times, pipette's max volume, in current location

Air Gap

Some liquids need an extra amount of air in the pipette’s tip to prevent it from sliding out. A call to air_gap() with a microliter amount will aspirate that much air into the tip.

pipette.aspirate(100, plate.wells("B4"))
pipette.air_gap(20)
pipette.drop_tip()

from opentrons import labware, instruments, robot

'''
Examples in this section expect the following
'''
tiprack = labware.load("opentrons_96_tiprack_300ul", "1")
plate = labware.load("96-flat", "2")

pipette = instruments.P300_Single(mount="right", tip_racks=[tiprack])

Controlling Speed

You can change the speed at which you aspirate or dispense liquid by either changing the defaults in the pipette constructor (more info under the Creating a Pipette section) or using our set_flow_rate function. This can be called at any time during the protocol.

from opentrons import labware, instruments, robot

'''
Examples in this section expect the following
'''
tiprack = labware.load("opentrons_96_tiprack_300ul", "1")
plate = labware.load("96-flat", "2")

pipette = instruments.P300_Single(mount="right", tip_racks=[tiprack])

pipette.set_flow_rate(aspirate=50, dispense=100)

You can also choose to only update aspirate OR dispense depending on the application. Pipette liquid handling speed is in ul/s.

Note The dispense speed also controls the speed of blow_out.

Moving

Move To

Pipette’s are able to move_to() any location on the deck.

For example, we can move to the first tip in our tip rack:

pipette.move_to(tiprack.wells("A1"))

You can also specify at what height you would like the robot to move to inside of a location using top() and bottom() methods on that location.

pipette.move_to(plate.wells("A1").bottom())  # move to the bottom of well A1
pipette.move_to(plate.wells("A1").top())     # move to the top of well A1
pipette.move_to(plate.wells("A1").bottom(2)) # move to 2mm above the bottom of well A1
pipette.move_to(plate.wells("A1").top(-2))   # move to 2mm below the top of well A1

The above commands will cause the robot’s head to first move upwards, then over to above the target location, then finally downwards until the target location is reached. If instead you would like the robot to move in a straight line to the target location, you can set the movement strategy to "direct".

pipette.move_to(plate.wells("A1"), strategy="direct")

Note

Moving with strategy="direct" will run the risk of colliding with things on your deck. Be very careful when using this option.

Usually the strategy="direct" option is useful when moving inside of a well. Take a look at the below sequence of movements, which first move the head to a well, and use “direct” movements inside that well, then finally move on to a different well.

pipette.move_to(plate.wells("A1"))
pipette.move_to(plate.wells("A1").bottom(1), strategy="direct")
pipette.move_to(plate.wells("A1").top(-2), strategy="direct")
pipette.move_to(plate.wells("A1"))

Delay

To have your protocol pause for any given number of minutes or seconds, simply call delay() on your pipette. The value passed into delay() is the number of minutes or seconds the robot will wait until moving on to the next commands.

pipette.delay(seconds=2)             # pause for 2 seconds
pipette.delay(minutes=5)             # pause for 5 minutes
pipette.delay(minutes=5, seconds=2)  # pause for 5 minutes and 2 seconds