Reference
Describing an Experiment
Experiment description file
All experiments are described by a file with the name
_ibl_experiment.description.yaml
. This description file contains
details about the experiment such as, information about the devices used
to collect data, or the behavior tasks run during the experiment. The
content of this file is used to copy data from the acquisition computer
to the lab server and also determines the task pipeline that will be
used to extract the data on the lab servers. It’s accuracy in fully
describing the experiment is, therefore, very important!
Here is an example of a complete experiment description file for a
mesoscope experiment running two consecutive tasks,
biasedChoiceWorld
followed by passiveChoiceWorld
.
devices:
mesoscope:
mesoscope:
collection: raw_imaging_data*
sync_label: chrono
cameras:
belly:
collection: raw_video_data
sync_label: audio
width: 640
height: 512
fps: 30
left:
collection: raw_video_data
sync_label: audio
right:
collection: raw_video_data
sync_label: audio
procedures:
- Imaging
projects:
- ibl_mesoscope_active
sync:
nidq:
acquisition_software: timeline
collection: raw_sync_data
extension: npy
tasks:
- _biasedChoiceWorld:
collection: raw_task_data_00
sync_label: bpod
extractors: [TrialRegisterRaw, ChoiceWorldTrialsTimeline, TrainingStatus]
- passiveChoiceWorld:
collection: raw_task_data_01
sync_label: bpod
extractors: [PassiveRegisterRaw, PassiveTaskTimeline]
version: 1.0.0
Breaking down the components of an experiment description file
Devices
The devices section in the experiment description file lists the set of devices from which data was collection in the experiment. Supported devices are Cameras, Microphone, Mesoscope, Neuropixel, Photometry and Widefield.
The convention for this section is to have the device name followed by a list of sub-devices, e.g.
devices:
cameras:
belly:
collection: raw_video_data
sync_label: audio
width: 640
height: 512
fps: 30
left:
collection: raw_video_data
sync_label: audio
right:
collection: raw_video_data
sync_label: audio
In the above example, cameras
is the device and the sub-devices are
belly
, left
and right
.
If there are no sub-devices, the sub-device is given the same name as the device, e.g.
devices:
mesoscope:
mesoscope:
collection: raw_imaging_data*
sync_label: chrono
Each sub-device must have at least the following two keys - collection - the folder containing the data - sync_label - the name of the common ttl pulses in the channel map used to sync the timestamps
Additional keys can also be specified for specific extractors, e.g. for the belly camera the camera metadata passed into the camera extractor task is defined in this file.
Procedures
The procedures section lists the set of procedures that apply to this experiment. The list of possible procedures can be found here.
As many procedure that apply to the experiment can be added e.g.
procedures:
- Fiber photometry
- Optical stimulation
- Ephys recording with acute probe(s)
Projects
The projects section lists the set of projects that apply to this experiment. The list of possible projects can be found here.
As many projects that apply to the experiment can be added e.g.
projects:
- ibl_neuropixel_brainwide_01
- carandiniharris_midbrain_ibl
Sync
The sync section contains information about the device used to collect the syncing data and the format of the data. Supported sync devices are bpod, nidq, tdms, timeline. Only one sync device can be specified per experiment description file and act as the main clock to which other timeseries are synced.
An example of an experiment run with bpod as the main syncing device is,
sync:
bpod:
collection: raw_behavior_data
extension: bin
Another example for spikeglx electrophysiology recordings with Neuropixel 1B probes use the nidq as main synchronisation.
sync:
nidq:
collection: raw_ephys_data
extension: bin
acquisition_software: spikeglx
Each sync device must have at least the following two keys - collection - the folder containing the data - extension - the file extension of the sync data
Optional keys include, for example acquisition_software
, the
software used to acquire the sync pulses
Tasks
The tasks section contains a list of the behavioral protocols run during the experiment. The name of the protocol must be given in the list e.g.
tasks:
- _biasedChoiceWorld:
collection: raw_task_data_00
sync_label: bpod
extractors: [TrialRegisterRaw, ChoiceWorldTrialsTimeline, TrainingStatus]
- passiveChoiceWorld:
collection: raw_task_data_01
sync_label: bpod
extractors: [PassiveRegisterRaw, PassiveTaskTimeline]
Each task must have at least the following two keys - collection - the folder containing the data - sync_label - the name of the common ttl pulses in the channel map used to sync the timestamps
The collection
must be unique for each task. i.e. Data from two
tasks cannot be stored in the same folder.
If the Tasks used to extract the data are not the default tasks, the extractors to use must be passed in as an additional key. The order of the extractors defines their parent child relationship in the task architecture.
Version
The version section gives version number of the experiment description file
Quality check the task post-usage
Once a session is acquired, you can verify whether the trials data is extracted properly and that the sequence of events matches the expected logic of the task.
Metrics definitions
All the metrics computed as part of the Task logic integrity QC (Task QC) are implemented in ibllib. When run at a behavior rig, they are computed using the Bpod data, without alignment to another DAQ’s clock.
Tip
The Task QC metrics definitions can be found in this documentation page. See this page on how to write QC checks for a custom task protocol.
Some are essential, i.e. if they fail you should immediately take action and verify your rig, and some are not as critical.
Essential taskQCs:
check_audio_pre_trial
check_correct_trial_event_sequence
check_error_trial_event_sequence
check_n_trial_events
check_response_feedback_delays
check_reward_volume_set
check_reward_volumes
check_stimOn_goCue_delays
check_stimulus_move_before_goCue
check_wheel_move_before_feedback
check_wheel_freeze_during_quiescence
Non essential taskQCs:
check_stimOff_itiIn_delays
check_positive_feedback_stimOff_delays
check_negative_feedback_stimOff_delays
check_wheel_move_during_closed_loop
check_response_stimFreeze_delays
check_detected_wheel_moves
check_trial_length
check_goCue_delays
check_errorCue_delays
check_stimOn_delays
check_stimOff_delays
check_iti_delays
check_stimFreeze_delays
check_wheel_integrity
Tip
The value returned by each metric is the proportion of trial that fail to pass the given test.
For example, if the value returned by check_errorCue_delays
is 0.92, it means 8% of the trials failed this test.
Quantifying the task QC outcome at the session level
The criteria for whether a session passes the Task QC is:
NOT_SET
: default value (= not run yet)FAIL
: if at least one metric is < 95%WARNING
: if all metrics are >=95% , and at least one metric is <99 %PASS
: if all metrics are >= 99%
This aggregation is done on all metrics, regardless if they are essential or not.
The criteria is defined at this code line
How to check the task QC outcome
Immediately after acquiring a session
At the behaviour PC, before the data have been copied, use the task_qc command with the session path:
task_qc C:\iblrigv8_data\Subjects\KS022\2019-12-10\001 --local
More information can be found here, or by running task_qc –help.
Once the session is registered on Alyx
Check on the Alyx webpage
From the session overview page on Alyx, find your session click on
See more session info
. The session QC is displayed in one of the right panels.To get more information regarding which test pass or fail (contributing to this overall session QC), you can click on the
QC
menu on the left. Bar-diagrams will appear, with essentials QCs on the left, colored in green if passing.Tip
You can hover over the bars with your mouse to easily know the name of the corresponding metric. This is useful if the value of the metric is
0
.Warning
If an essential metric fails, run the Task QC Viewer to investigate why.
Run the taskQC Viewer to investigate
The application Task QC Viewer enables to visualise the data streams of problematic trials.
Tip
Unlike when run at the behaviour PC, after registration the QC is run on the final time-aligned data (if applicable).
Run the task QC metrics and viewer
Select the
eid
for your session to inspect, and run the following within the iblrig env:task_qc baecbddc-2b86-4eaf-a6f2-b30923225609
Guide to develop a custom task
iblrigv8 design: inheritance
During the lifetime of the IBL project, we realized that multiple task variants combine with multiple hardware configurations and acquisition modalities, leading to a combinatorial explosion of possible tasks and related hardware.
This left us with the only option of developing a flexible task framework through hierarchical inheritance.
- All tasks inherit from the
iblrig.base_tasks.BaseSession
class, which provides the following functionalities: read hardware parameters and rig parameters
optionally interfaces with the Alyx experimental database
creates the folder structure for the session
writes the task and rig parameters, log, and acquisition description files
Additionally the iblrig.base_tasks
module provides “hardware mixins”. Those are classes that provide hardware-specific functionalities, such as connecting to a Bpod or a rotary encoder. They are composed with the BaseSession
class to create a task.
Warning
This sounds complicated ? It is ! Forecasting all possible tasks and hardware add-ons and modification is fool’s errand, however we can go through specific examples of task implementations.
Guide to Creating Your Own Task
What Happens When Running an IBL Task?
The task constructor is invoked, executing the following steps:
Reading of settings: hardware and IBLRIG configurations.
Reading of task parameters.
Instantiation of hardware mixins.
The task initiates the
run()
method. Prior to execution, this method:Launches the hardware modules.
Establishes a session folder.
Saves the parameters to disk.
The experiment unfolds: the
run()
method triggers the_run()
method within the child class:Typically, this involves a loop that generates a Bpod state machine for each trial and runs it.
Upon SIGINT or when the maximum trial count is reached, the experiment concludes. The end of the
run()
method includes:Saving the final parameter file.
Recording administered water and session performance on Alyx.
Halting the mixins.
Initiating local server transfer.
Examples
Where to write your task
After the installation of iblrig the project extraction repository is located at the root of the C: drive.
New tasks should be added to the C:\project_extraction\iblrig_custom_tasks
folder to be made visible by the iblrig GUI.
We use a convention that the task name starts with the author identifier, followed by an underscore, followed by the task name, such as olivier_awesomeChoiceWorld.
- olivier_awesomeChoiceWorld
__init__.py
task.py
README.md
task_parameters.yaml
test_olivier_awesomeChoiceWorld.py
Example 1: variation on biased choice world
We will create a a choice world task that modifies a the quiescence period duration random draw policy.
In the task.py file, the first step is to create a new task class that inherits from the BiasedChoiceWorldSession
class.
Then we want to make sure that the task bears a distinctive protocol name, _iblrig_tasks_imagingChoiceWorld. We also create the command line entry point for the task that will be used by the iblrig GUI.
Also, in this case we can leverage the IBL infrastructure to perform extraction of the trials using existing extractors extractor_tasks = [‘TrialRegisterRaw’, ‘ChoiceWorldTrials’]
import iblrig.misc from iblrig.base_choice_world import BiasedChoiceWorldSession class Session(BiasedChoiceWorldSession): protocol_name = "_iblrig_tasks_imagingChoiceWorld" def __init__(self, *args, **kwargs): self.extractor_tasks = ['TrialRegisterRaw', 'ChoiceWorldTrials'] super().__init__(*args, **kwargs) if __name__ == "__main__": # pragma: no cover kwargs = iblrig.misc.get_task_arguments(parents=[Session.extra_parser()]) sess = Session(**kwargs) sess.run()
In this case the parent class BiasedChoiceWorldSession has a method that draws the quiescence period. We are going to overload this method to add our own policy. This means the parent method will be fully replaced by our implementation. The class now looks like this:
class Session(BiasedChoiceWorldSession): protocol_name = "_iblrig_tasks_imagingChoiceWorld" def draw_quiescent_period(self): """ For this task we double the quiescence period texp draw and remove the absolute offset of 200ms. The resulting is a truncated exp distribution between 400ms and 1 sec """ return iblrig.misc.texp(factor=0.35 * 2, min_=0.2 * 2, max_=0.5 * 2)
Et voilà, in a few lines, we re-used the whole biased choice world implementation to add a custom parameter. This is the most trivial and easy example. The full code is available here.
Example 2: re-writing a state-machine for a biased choice world task
In some instances changes in the task logic require to go deeper and re-write the sequence of task events. In bpod parlance, we are talking about rewritng the state-machine code.
Coming, for now here is an example of such a task.
Packaging
You can package your task as a plugin for IBLRIG that resides in an independant package. For instance, if your package is structured as follows:
example_task
├── pyproject.toml
├── task_parameters.yaml
└── task.py
include the following configuration in your pyproject.toml
file:
[project]
name = "example_task"
version = "0.1.0"
description = "An example task for IBLRIG"
[project.entry-points."iblrig.plugins"]
task_example = "task:Session"
In order for IBLRIG to recognize your plugin, the key of the entry point (in this case, task_example
) must start with the prefix task_
. The value must point to a subclass of BaseSession
, which is the base class for all tasks in IBLRIG.
Once you have created your package and installed it in the virtual environment of IBLRIG, your task should appear in the list of available tasks in the GUI.