This is where Opentrons protocols are stored for everyone to use.
The master
branch populates http://protocols.opentrons.com/, our Protocol Library. Please let us know if you would like to contribute your protocols, or just submit a pull request. We would love to add your Opentrons protocols to the Library!
All the best,
Will Canine Co-Founder, Opentrons [email protected]
The develop
branch populates the staging version of the Opentrons Protocol Library at http://develop.protocols.opentrons.com/. Pull requests should be made to develop
to be staged, and we will merge the changes into master
as a second step.
Every protocol needs its own folder. In that folder, there needs to be:
- A single
README.md
readme file - A single
.py
Opentrons protocol file - Optional "dot files" (see below)
Every protocol should have a README file in its folder, with the file name README.md
. It is a Markdown file, with a specific format:
# Human-Readable Protocol Name Here
### Author
[Your Name here](http://www.your.web.site.com/)
### Partner
[Partner Name here](https://www.example.com/)
## Categories
* Proteins & Proteomics
* Assay
## Description
Describe and summarize the protocol here.
What is the purpose of the protocol?
What special labware is required?
### Time Estimate
30 minutes
### Robot
* [OT PRO](https://opentrons.com/ot-one-pro)
* [OT Standard](https://opentrons.com/ot-one-standard)
* [OT Hood](https://opentrons.com/ot-one-hood)
### Modules
* [CoolDeck](https://shop.opentrons.com/collections/labware/products/cold-deck)
### Reagents
* ddH2O
* cDNA samples
* Et cetera
## Process
1. Describe the steps that a scientist takes to execute the protocol.
2. Put the steps in a numbered list
3. Et cetera
### Additional Notes
Notes here will show up on the bottom of the protocol on the website.
###### Internal
Notes here will not be published on the Opentrons Protocol Library website,
but will be visible in the public GitHub Protocols repo.
"Dot files" are files inside a protocol folder which start with a dot (.
).
These files are usually blank text files. They have special names that indicate specific properties:
.feature
- The protocol will be listed under "Featured Protocols" on the website..ignore
- The protocol will not be shown on the Opentrons Protocol Library, even if you search for it..notests
- The protocol will not be tested by continuous integration. This is intended only for ignored protocols..embedded
- This is for "embedded apps" that generate a protocol and are designed to be shown in the Protocol Library in an iframe. This file should not be blank, it should contain a URL to the web app that will be embedded in the iframe.
To avoid waiting for delays while simulating protocols during the build, Pipette.delay
, Magbead.delay
, and time.sleep
are monkeypatched so that they always delay for 0 seconds during the build. They will work normally when the protocol is executed by the app.
If there are any cases that the monkeypatched delays don't cover, please raise an issue on GitHub. In the meantime, you can use the OT_TESTING
environment variable -- anything truthy should indicate that a script is running in a testing/simulation environment:
# Bypass time.sleep in testing env
import os
hours_to_incubate = 8
if os.getenv('OT_TESTING') is not None:
hours_to_incubate = 0
myPipette.delay(60*60*hours_to_incubate)
"Custom protocols" allow users to set variables for their protocol on the protocol library site before downloading a protocol,
Custom protocols is an early-stage feature, under active development. They are subject to change.
First, set up your containers and instruments normally at the top of the .py
file.
For some protocols, you might want "number of destination plates" to be a variable. However, the deck map on the website is currently not dynamic - it will only show containers loaded at the top of the file. For this reason, you should load all the containers you might need at the top of the file, and then only use what you need during the actual execution.
To make a protocol customizable, put all your commands that run on the robot in a function called run_custom_protocol
.
The arguments to that function will be used to create input forms on the Protocol Library website page for your protocol.
You can use Python function annotations to specify what type of input to use. Right now, only float
and int
are supported.
Field names on the protocol's webpage will be named after the arguments. Eg, number_of_samples: int=2
becomes Number of samples
on the form, an integer field with a default of 2.
Form validation, such as setting min and max values, is not currently supported.
Inside your run_custom_protocol
function, write all your robot commands (transfer
, distribute
, etc.)
Use a multi-channel pipette to transfer a custom number of rows from one plate to another, with a custom transfer volume
from opentrons import instruments, containers
# set up containers and instruments
source = containers.load('96-flat', 'C1')
dest = containers.load('96-flat', 'E1')
trash = containers.load('trash-box', 'B2')
tiprack = containers.load('tiprack-200ul', 'A1')
p200_multi = instruments.Pipette(
axis='a',
trash_container=trash,
tip_racks=[tiprack],
max_volume=200,
min_volume=20,
channels=8,
)
# set up special `run_custom_protocol` function, with annotated arguments
def run_custom_protocol(transfer_volume: float=1.0, number_of_rows: int=1):
# all commands go in this function
p200_multi.distribute(
transfer_volume,
source.rows(0, to=number_of_rows),
dest.rows(0, to=number_of_rows))
In this repo, the otcustomizers
python module contains some utilities that you can use to
generate "customization widgets" like dropdown menus and file upload fields on your protocol's
page on the Protocol Library.
To install the otcustomizer module, open a terminal, cd
into the root directory of this repo,
and do pip install -e otcustomizers
.
Now you can do from otcustomizers import StringSelection, FileInput
, etc, and use
these imports in your protocol.
If you forget to install otcustomizers
, you will get the error:
ModuleNotFoundError: No module named 'otcustomizers'
A StringSelection
argument to your run_custom_protocol
function will create a dropdown menu
on your protocol's page on the Protocol Library website.
Then use it in your run_custom_protocol
function:
from otcustomizers import StringSelection
# maybe some setup stuff here...
def run_custom_protocol(
well_volume: float=20.0,
plate_type: StringSelection('96-flat', '96-PCR-tall', '96-deep-well')='96-flat',
tuberack_type: StringSelection('tube-rack-.75ml', 'tube-rack-2ml')='tube-rack-.75ml'):
plate = containers.load(plate_type, 'A1')
tube_rack = containers.load(tuberack_type, 'C1')
# do stuff with the plate here...
The line plate_type: StringSelection('96-flat', '96-PCR-tall', '96-deep-well')='96-flat'
means:
- Create a dropdown selection menu called "Plate Type" with the options: '96-flat', '96-PCR-tall', '96-deep-well'
- The default value will be '96-flat' (from the
='96-flat'
at the end)
And another selector menu is made for "Tuberack Type" with options: 'tube-rack-.75ml', 'tube-rack-2ml'.
A FileInput
argument to your run_custom_protocol
function creates a file upload button
on your protocol's page on the Protocol Library website.
The uploaded file will be passed into the run_custom_protocol
function as plain text.
If you want to parse a .csv
(comma separated values) text file, you can copy and paste
the well_csv_to_list
helper below.
from otcustomizers import FileInput
# maybe some setup stuff here...
def well_csv_to_list(csv_string):
"""
Takes a csv string and flattens it to a list, re-ordering to match
Opentrons API well order convention (A1, B1, C1, ..., A2, B2, B2, ...).
The orientation of the CSV cells should match the "portrait" orientation
of plates on the OT-One: well A1 should be on the bottom left cell. Example:
...
A3, B3, C3, ...
A2, B2, C2, ...
A1, B1, C1, ...
Returns a list: [A1, B1, C1, ..., A2, B2, C3, ...]
where each CSV cell is a string in the list.
"""
return [
cell
for line in reversed(csv_string.split('\n')) if line.strip()
for cell in line.split(',') if cell
]
# Don't forget to include a default file string,
# so that unit tests which call run_custom_protocol()
# with its default arguments will pass.
example_csv = """
50,50,50,50,50,50,50,50
50,50,50,50,50,50,50,50
50,50,50,50,50,50,50,50
50,50,50,50,50,50,50,50
50,50,50,50,50,50,50,50
50,50,50,50,50,50,50,50
50,50,50,50,50,50,50,50
50,50,50,50,50,50,50,50
50,50,50,50,50,50,50,50
50,50,50,50,50,50,50,50
50,50,50,50,50,50,50,50
50,50,50,50,50,50,50,50
"""
def run_custom_protocol(volumes_csv: FileInput=example_csv):
# `example_csv` will be a multi-line string containing the contents of the uploaded file
# use the helper csv function to parse that string
csv_list = well_csv_to_list(volumes_csv)
# convert the cell contents from strings to floats
volumes = [float(cell) for cell in csv_list]
pipette.transfer(volumes, source, plate)