- 🗞️ Overview
- 🗝️ Quick start
- 🚀 How to run
- 🌻 Acknowledgement
- 🔖 Citation
InstructCell is a multi-modal AI copilot that integrates natural language with single-cell RNA sequencing data, enabling researchers to perform tasks like cell type annotation, pseudo-cell generation, and drug sensitivity prediction through intuitive text commands. By leveraging a specialized multi-modal architecture and our multi-modal single-cell instruction dataset, InstructCell reduces technical barriers and enhances accessibility for single-cell analysis.
InstructCell has two versions:
- Chat Version: Supports generating both detailed textual answers and single-cell data, offering comprehensive and context-rich outputs.
- Instruct Version: Supports generating only the answer portion without additional explanatory text, providing concise and task-specific outputs.
Both versions of the model are available for download from Hugging Face (zjunlp/InstructCell-chat and zjunlp/InstructCell-instruct).
- python 3.10 and above are recommended
- CUDA 11.7 and above are recommended
We provide a simple example for quick reference. This demonstrates a basic cell type annotation workflow.
Make sure to specify the paths for H5AD_PATH
and GENE_VOCAB_PATH
appropriately:
H5AD_PATH
: Path to your.h5ad
single-cell data file (e.g.,H5AD_PATH = "path/to/your/data.h5ad"
).GENE_VOCAB_PATH
: Path to your gene vocabulary file (e.g.,GENE_VOCAB_PATH = "path/to/your/gene_vocab.npy"
).
from mmllm.module import InstructCell
import anndata
import numpy as np
from utils import unify_gene_features
# Load the pre-trained InstructCell model from HuggingFace
model = InstructCell.from_pretrained("zjunlp/InstructCell-chat")
# Load the single-cell data (H5AD format) and gene vocabulary file (numpy format)
adata = anndata.read_h5ad(H5AD_PATH)
gene_vocab = np.load(GENE_VOCAB_PATH)
adata = unify_gene_features(adata, gene_vocab, force_gene_symbol_uppercase=False)
# Select a random single-cell sample and extract its gene counts and metadata
k = np.random.randint(0, len(adata))
gene_counts = adata[k, :].X.toarray()
sc_metadata = adata[k, :].obs.iloc[0].to_dict()
# Define the model prompt with placeholders for metadata and gene expression profile
prompt = (
"Can you help me annotate this single cell from a {species}? "
"It was sequenced using {sequencing_method} and is derived from {tissue}. "
"The gene expression profile is {input}. Thanks!"
)
# Use the model to generate predictions
for key, value in model.predict(
prompt,
gene_counts=gene_counts,
sc_metadata=sc_metadata,
do_sample=True,
top_p=0.95,
top_k=50,
max_new_tokens=256,
).items():
# Print each key-value pair
print(f"{key}: {value}")
For more detailed explanations and additional examples, please refer to the Jupyter notebook demo.ipynb.
Assume your current directory path is DIR_PATH
.
The datasets used in the paper are all publicly available.
Detailed instructions and dataset links are provided in the Jupyter notebooks: HumanUnified.ipynb
and MouseUnified.ipynb
. Below is a summary of the datasets and their corresponding details:
🔗 Please Note:
For the He-2020 dataset, the cell type annotation file is sourced from the GitHub repository scRNA-AHCA 👈.
Follow these steps to set up InstructCell:
- Clone the repository:
git clone https://github.com/zjunlp/InstructCell.git
- Set up a virtual environment and install the dependencies:
conda create -n instructcell python=3.10
conda activate instructcell
cd InstructCell
pip install -r requirements.txt
The pre-trained language model used in this project is T5-base. You can download it from 🤗 Hugging Face and place the corresponding model directory under DIR_PATH
.
Alternatively, you can use the provided script to automate the download process:
python download_script.py --repo_id google-t5/t5-base --parent_dir ..
Navigate to the parent directory DIR_PATH
and organize your data by creating a main data folder and three task-specific subfolders:
cd ..
mkdir data
cd data
mkdir cell_type_annotation
mkdir drug_sensitivity_prediction
mkdir conditional_pseudo_cell_generation
cd ..
For dataset preprocessing, refer to the previously mentioned Jupyter notebooks:
- HumanUnified.ipynb for human datasets.
- MouseUnified.ipynb for mouse datasets.
Note
Matching orthologous genes between mouse and human is based on pybiomart and pyensembl. Before preprocessing mouse datasets, ensure the corresponding Ensembl data are downloaded by running:
pyensembl install --release 100 --species mus_musculus
After completing the preprocessing steps, split each dataset and build a gene vocabulary using the following command:
cd InstructCell
python preprocess.py --n_top_genes 3600
To customize the size of the gene vocabulary, adjust the n_top_genes
parameter as needed. For instance, setting it to 2000 will generate a smaller vocabulary. At this point, two files, gene_vocab.npy
and choices.pkl
, are generated. The first file stores the selected genes, while the second holds the category labels for each classification dataset. The gene vocabulary and label set used in this project can both be found in this folder.
The instruction-response templates used in the projects are stored in this folder.
The construction of instruction-response templates is divided into four stages:
- Motivation and personality generation: In this stage, the large language model is prompted to generate potential motivations for each task and corresponding personalities. This step is implemented in the
data_synthesis.py
script. - Template synthesis via parallel API calls: Multiple APIs are run in parallel to synthesize templates, with each API invoked a specified number of times per task. This process is also implemented in the
data_synthesis.py
script. - Merging synthesized templates: The generated templates are consolidated into a unified collection using the
merge_templates.py
script. - Filtering and splitting templates: Finally, the templates are filtered for quality and divided into specific datasets using the
split_templates.py
script.
To execute all four stages in sequence, use the run_data_synthesis.sh
script:
bash run_data_synthesis.sh
Note
Before executing run_data_synthesis.sh
, ensure the parameters in the script are configured correctly. Update the API keys and base URL as needed, specify the model for template synthesis (model
in the script), and adjust the number of API calls per task (num_templates_for_task
in the script).
To train InstructCell, use the following command:
torchrun --nproc_per_node=8 mm_train.py \
--epochs 160 \
--save_freq 20 \
--batch_size 64 \
--train_template_dir ../output/train_templates \
--valid_template_dir ../output/valid_templates \
--save_dir ../checkpoints \
--best_model_dir ../trained_models \
--train_no_extra_output_ratio 1.0 \
--eval_no_extra_output_ratio 1.0
- To obtain the chat version of InstructCell, set both
train_no_extra_output_ratio
andeval_no_extra_output_ratio
to 0. - To resume training from a specific checkpoint (
YOUR_CHECKPOINT_PATH
), include the flags--resume True
and--resume_path YOUR_CHECKPOINT_PATH
. - For training on a single task and dataset, modify the
TASKS
parameter inmetadata.py
, retain only one dataset directory in the corresponding task folder, and set--unify_gene False
. - You can customize the architecture of InstructCell (e.g., the number of query tokens in Q-Former or the latent variable dimensions in the VAE) by modifying the
MODEL_PARAMETERS
inmetadata.py
.
To evaluate the performance of InstructCell on conditional pseudo-cell generation, run:
python evaluate.py \
--best_model_path ../trained_models/best_mm_model.pkl \
--task_type "conditional pseudo cell generation" \
--template_dir_name ../output/test_templates \
--no_extra_output_ratio 1.0
- To evaluate InstructCell on other tasks, modify the
task_type
parameter to"cell type annotation"
or"drug sensitivity prediction"
accordingly. - To test InstructCell’s robustness to different task descriptions, add the flag
--evaluate_single_prompt True
. By default, 20 different task descriptions are used. To increase this number (e.g., to 40), include--num_single_prompt 40
. - If you want to evaluate only test templates that contain options, add
--provide_choices True
. By default, all test templates are evaluated. - To evaluate the chat version of InstructCell, set the
no_extra_output_ratio
parameter to 0.0. This will generate content formatted for xFinder’s JSON input requirements. For detailed evaluation procedures using xFinder, please visit the xFinder repository 👈. Alternatively, you can refer to the README provided in our repository for additional guidance.
We would like to express our sincere gratitude to the excellent work ALBEF, and scvi-tools.
If you use the code or data, please cite the following paper:
@article{fang2025instructcell,
title={A Multi-Modal AI Copilot for Single-Cell Analysis with Instruction Following},
author={Fang, Yin and Deng, Xinle and Liu, Kangwei and Zhang, Ningyu and Qian, Jingyang and Yang, Penghui and Fan, Xiaohui and Chen, Huajun},
journal={arXiv preprint arXiv:2501.08187},
year={2025}
}
We will offer long-term maintenance to fix bugs and solve issues. So if you have any problems, please put issues to us.