HMMSTR calls tandem repeat copy number from raw, long-read, sequencing reads and reports copy numbers in both a read and sample specific format. While designed to model Nanopore sequencing errors in repetitive regions, HMMSTR can be applied to PacBio data as well and has flexible arguments to allow for custom error rates.
HMMSTR is optimized for targeted sequencing experiments and can be run with a single or multiple target regions/sequences in a global-alignment and reference free format.
Kinsey Van Deynze, Camille Mumm, Connor J Maltby, Jessica A Switzenberg, Peter K Todd, Alan P Boyle, Enhanced detection and genotyping of disease-associated tandem repeats using HMMSTR and targeted long-read sequencing, Nucleic Acids Research, 2024;, gkae1202, https://doi.org/10.1093/nar/gkae1202
-
Python >= 3.8.17, < 3.12
-
colorama
-
numpy
-
pandas
-
pickleshare
-
scikit-learn
-
scipy
-
seaborn
-
importlib-resources
-
mappy
-
glib-2 (Ubuntu install command below)
sudo apt install libglib2.0-dev
HMMSTR is currently available on Pypi and Anaconda
pip install HMMSTR
conda install kvandeynze::hmmstr
A small test fasta file is included under tests. Test your install with:
hmmstr targets_tsv tests/AAAAG_input.txt test_install tests/AAAAG_11012021_10.fa --max_peaks 3
usage: hmmstr [-h] [--output_plots] [--output_labelled_seqs] [--max_peaks MAX_PEAKS] [--cpus CPUS] [--flanking_size FLANKING_SIZE] [--mode MODE] [--mapq_cutoff MAPQ_CUTOFF] [--k K] [--w W] [--use_full_read] [--peakcalling_method PEAKCALLING_METHOD] [--bandwidth BANDWIDTH] [--kernel KERNEL] [--bootstrap] [--call_width CALL_WIDTH] [--resample_size RESAMPLE_SIZE] [--allele_specific_CIs] [--allele_specific_plots]
[--discard_outliers] [--filter_quantile FILTER_QUANTILE] [--flanking_like_filter] [--stranded_report] [--cluster_only] [--save_intermediates] [--background BACKGROUND] [--E_probs E_PROBS] [--A_probs A_PROBS] [--custom_RM CUSTOM_RM] [--hmm_pre HMM_PRE]
{targets_tsv,coordinates} ... out inFile
HMMSTR has 2 input modes:
Directly uses an input tsv with the following columns:
name
: the names of all targets for a given runprefix
: the sequence directly upstream of the target repeat (200bp recommended)repeat
: repeat motif for given target (must be on the same strand as prefix and suffix sequences)suffix
: the sequence directly downstream of the target repeat (200bp recommended)
Uses a custom bedfile to create the targets tsv from the following columns:
Chromosomes
Start coordinate
End coordinate
Repeat motif
(on same strand as reference genome used)Target name
(optional, if not given name will be assigned as chr:start-end)
coordinates also requires the following additional positional arguments:
chrom_sizes
: path to chromosome sizes file corresponding to the reference genome usedref
: path to the reference genome to get flanking sequences frominput_flank_length
: Length of the prefix and suffix to get from the reference genome, must be longer than 100bp (Default) or the--flanking_size
optional parameter, (optional, default: 200)
Optionally, the user may also input all options as a text file which each input parameter and option on its own line. An example of this can be found here. This file is also automatically output to [out_prefix]_run_input.txt as a record of the run.
Argument | Description |
---|---|
out | Path to output directory with prefix for all results in this run |
inFile | Sequence file in fasta or fastq format, can be gzipped (must end with .gz). BAM input supported for coordinates input mode only (must end with .bam). |
Optional Arguments
Argument | Description |
---|---|
--cpus | Maximum number of CPUs to use during read processing step (default: half of available CPUs) |
--use_full_read | If passed, HMMSTR will use the full read sequence to predict copy number instead of subsetting each read based on flanking sequence alignment. Optimal for runs where the repeat is close to the end or start of the reads consistently (ie when running on PCR products where primers are relatively close to the repeat of interest) |
Argument | Description |
---|---|
--flanking_size | Integer designating the number of bases flanking the repeat to encode in the model. Must be shorter or equal in length to given prefix and suffix. Note: significant increases in flanking size will increase runtime but may increase accuracy in low complexity regions. Longer flanking sequences are recommended for regions with high similarity with respect to sequence directly flanking the repeat (default: 100, 100-200 recommended for highly repetitive regions, 30 for increased speed) |
Argument | Description |
---|---|
--mode | Mode used by mappy. map-ont (Nanopore), pb (PacBio), or sr (short accurate reads, use for short flanking sequence input) (default: map-ont) |
--mapq_cutoff | MapQ cutoff for prefix and suffix (default: 30, range: 0-60) |
Argument | Description |
---|---|
--max_peaks | Integer designating the maximum number of alleles to call for a given run (default: 2) |
--peakcalling_method | Used to override the default peak calling pipeline. Options include: gmm, kde, kde_throw_outliers (default:auto, HMMSTR chooses the best method based on the distribution of copy numbers per target) |
--discard_outliers | If passed, outliers per read-level copy number will be discarded based on quantile. If --filter_quantile not set, reads exceding the top and bottom quantile (0.25) will be discarded and marked as outliers in outputs |
--filter_quantile | Float designating quantile of count frequency to discard when filtering outliers (default: 0.25) |
--flanking_like_filter | If passed, outliers determined by the likelihood of the flanking sequence will be filtered. This is an additional filter for off-targets or low quality reads |
Argument | Description |
---|---|
--bandwidth | Bandwidth to use for KDE. It is recommended to use the default scott method, especially when there is no expectation for the distribution of repeat lengths. |
--kernel | Kernel to use for the KDE. Default is gaussian, allows for other kernels if testing different distributions is desired. |
Argument | Description |
---|---|
--output_plots | output supporting reads histogram showing how many reads were assigned to each repeat copy number per target in a single run as well as the model of best fit |
--bootstrap | Boolean designating to output bootstraped confidence intervals for allele calls. By default, the samples are drawn from the full dataset regardless of allele. |
--output_labelled_seqs | Output the model path through prefix, repeat, and suffix identified per read as context_labelled.txt per target. This is useful for inspecting repeat sequences as well as how well your model fit your target of interest. |
--stranded_report | If set, genotypes are called for each strand separately and strand bias is reported if found. |
Argument | Description |
---|---|
--call_width | Decimal percentage designating confidence interval width to calculate in bootstrapping (default: 0.95) |
--resample_size | Number of times to resample the repeat copy number distribution during bootstrapping (default:100) |
--allele_specific_CIs | Output allele-specific bootstrapped confidence intervals. This process separates data by assigned alleles before sampling. |
--allele_specific_plots | Output allele-specific histograms with model of best fit. Helpful when visualizing alleles with significantly different support |
Advanced Options
Optional tsv inputs to set custom model parameters.
Argument | Description |
---|---|
--background | TSV with custom background frequencies to encode in genome states (example here |
--E_probs | TSV with custom emission probabilities to be encoded in match states. These should correspond to the expected mismatch rate (example here) |
--A_probs | TSV with custom transition probibilities to be encoded in the model. Column names in "P_xy" format such that 'x' is the first state type and 'y' is the state type 'x' transitions to (example here) |
--custom_RM | TSV with columns corresponding to a given postion in the repeat motif and rows corresponding to possible nucleotides (and deletion character ''). This is used to designate custom nucleotide occupancy per position in a given motif in case of known mosaicism (ie AAGGG vs AAAAG at the CANVAS locus). Note: this matrix will be applied to all models in a given run, it is advised you only use it in single target runs (example here) |
Parameters to pass to Mappy during alignment step
Argument | Description |
---|---|
--k | Integer designating kmer parameter to be passed to mappy (see mappy documentation) |
--w | Window parameter to be passed to mappy (see mappy documentation) |
Parameters to use to test different clustering methods on your data
Argument | Description |
---|---|
--save_intermediates | Flag designating to save intermediate files including model inputs, raw count files, and state sequence files. NOTE: raw count files are required to recall alleles without rerunning the counting algorithm, see --cluster_only |
--cluster_only | Only run peak calling step on existing raw repeat copy counts data 'out''target_name'_counts.txt . NOTE: Must use the same output and target names as the run that produced the counts files. |
Output File Details
There are two tsv files output by HMMSTR by default, a description of the columns included in both are as follows:
This file has one row per target in the given input
name
: name of the target designated by inputA1:median
: median repeat copy number of allele oneA1:mode
: mode repeat copy number of allele oneA1:SD
: standard deviation of the allele one clusterA1:supporting_reads
: the number of reads assigned to allele 1num_supporting_reads
: total number of reads assigned to any allelebandwidth
: if KDE was used for peak calling, the bandwidth selected will display here, otherwise it is set to -1.0peak_calling_method
: which peak caller was used for a given target
Note: All allele specific columns will repeat up to the max_peaks
parameter set by input
This file has one row for every target a given read was assigned to, thus if a read is assigned to multiple targets it will show up multiple times
name
: name of target given read was assigned toread_id
: id of the readstrand
: the strand of the read relative to the input sequence or referencealign_score
: combined mapq of prefix and suffix sequencesneg_log_likelihood
: the negative-log-likelihood of the Viterbi path the read takes through the target model. Note: this is for the subsetted read in the default case, not the full read sequencesubset_likelihood
: the negative-log-likelihood of the sequence labelled as prefix, repeat, and suffix statesrepeat_likelihood
: the negative-log-likelihood of the identified repeat sequencerepeat_start
: the start index of the repeat relative to the full input read stringrepeat_end
: the end index of the repeat relative to the full input read stringalign_start
: the start index of the start of the upstream alignment (either prefix or suffix dependent on the strand)align_end
: the end index of the end of the downstream alignment (either prefix or suffix dependent on the strand)counts
: the number of repeat copies called in the given readfreq
: the frequency of the copy number for the assigned targetcluster_assignments
: which allele the given read was assigned to during peak callingoutlier
: boolean designating if the given read was discarded before peak calling due to exceeding the IQR of the data (if applicable, will always be False if --discard_outliers not passed)peak_calling_method
: the peak calling method used to assign the read to its allele
Basic Use: Single Plasmid Target
Here, we run HMMSTR on a sequence file containing nanopore reads from a plasmid construct with variable copies of an AAAAG repeat motif. Since these are plasmid contructs, we wrote our input tsv file AAAAG_input.txt by setting the prefix column to the 200bp upstream sequnce directly flanking the AAAAG repeat from the known backbone sequence and set the suffix column with the downstream flanking sequence. For this example, we will use all default parameters with the exception of --output_plots
, --max_peaks
, and --output_labelled_seqs
.
hmmstr targets_tsv AAAAG_input.txt ./tutorial_1 AAAAG_11012021_3000_sample.fasta --max_peaks 3 --output_plots --output_labelled_seqs
tutorial_1_genotype_calls.tsv
: TSV containing final allele calls per targettutorial_1_read_assignments.tsv
: TSV containing read level statistics and coordinates, copy number predictions, and allele assignmentstutorial_1_run_parameters.txt
: Text file with all parameters used in the run in "parameter : value" format including default values.tutorial_1_run_input.txt
: Text file with all inputs in the format compatible with running HMMSTR with a file input, that is, one input parameter per line in the same format as the command line version. This file can be used to reproduce the run or used as a record of the run.
The following are output to a directory with suffix "_labelled_seqs
AAAAG_context_labeled.txt
: (Optional) Text file contianing repeat sequence and flanking context sequence colored by the optimal state path along with the read name and strand. This can be viewed on the command line. This is helpful when determining if the prefix/suffix you inputted are well fit to the repeat of interest and can help in debugging your inputs. This file is produced for each input target.
The following are output to a directory with suffix "_plots"
tutorial_1AAAAGpeaks.pdf
: (Optional) Supporting read histogram displayed with the model of best fit as a density plot -- GMM or KDE depending on the peak caller chosen.tutorial_1AAAAGAIC_BIC.pdf
: (Optional) If GMM chosen, the AIC and BIC are plot and outputted here. These metrics are used to determine the most likely number of clusters.tutorial_1AAAAG_supporting_reads_hist.pdf
: (Optional) Raw supporting read histogram, copy number by number of supporting reads.
Below is an example of the *context_labeled.txt files:
- Red rectangles represent deletions, green represents insertions, bases labeled as in the repeat sequence are white and the prefix and suffix are in grey
Supporting read histogram Model of best fit -- GMM AIC/BIC plot
If the same command is run with the KDE --peakcalling_method
option, the model of best fit plot would be the following:
hmmstr targets_tsv AAAAG_input.txt ./tutorial_1 AAAAG_11012021_3000_sample.fasta --max_peaks 3 --output_plots --peakcalling_method kde
HMMSTR also includes options to visualize per-read copy number prediction distributions in an allele-specific format. Below is how we would use HMMSTR to output these plots as well as allele-specific confidence intervals. Note: these confidence intervals are produced by bootstrapping the median of a given allele with 100 resamples.
hmmstr targets_tsv AAAAG_input.txt ./tutorial_1 AAAAG_11012021_3000_sample.fasta --output_plots --max_peaks 3 --bootstrap --resample_size 100 --allele_specific_CIs --allele_specific_plots
Allele 1 | Allele 2 | Allele 3 |
---|---|---|
(30.0, 30.0) | (58.0, 59.0) | (16.0, 16.0) |
Repeat Expansion Panel
HMMSTR was designed as a companion tandem repeat caller for our repeat expansion panel as described in our manuscript. Below is an example of how to run one set of our targets in coordinates
.
Run with coordinates
input and all default parameters except --mapq_cutoff
(we want to be strict with reads we accept)
hmmstr coordinates $TARGET_COORDS $CHR_SIZES $REF $OUT $INFILE --mapq_cutoff 60
This run will also produce the accompanying input file for future target_tsv
runs under the output directory and prefix as _inputs.tsv
One caveat you may run into is exceedingly low (1-2 reads) or unbalanced coverage across expanded alleles in an expansion positive sample. In this case, HMMSTR may discard the expanded allele if either --discard_outliers
or --peakcalling_method kde_throw_outliers
are passed. To account for this, it is recommended that in these cases you do not use either of these modes but rather override the default peak caller as follows:
hmmstr coordinates $TARGET_COORDS $CHR_SIZES $REF $OUT $INFILE --mapq_cutoff 60 --peakcalling_method gmm
This will ensure the entire dataset is considered during genotyping. Note: this will also result in an increase of false heterozygous calls for homozygous regions. If you wish to have high accuracy for both expanded alleles and homozygotes, consider running HMMSTR with both settings on the same sequence file.
If there is sufficient coverage across all alleles in the run, this is not an issue.
FAQs and Common Use Cases
- Why use one peak calling method over another?
- Auto (default): The default peak caller will automatically designate a method per target based on the distribution of the data. This assumes that you have enough coverage across all alleles.
- KDE: The Kernel Density peakcaller differentiates heterozygous and homozygous alleles better than the GMM peak caller; however, it is more easily skewed by outliers if used without discarding outliers. KDE is also better at separating data into independent distributions in cases with high noise.
- GMM: The Gaussian mixture model peakcaller is more robust to outliers and uneven coverage across alleles. We recommend this option be used if you want higher sensitivity to detecting expanded alleles at low coverage and are less concerned about resolving heterozygous vs homozygous alleles with low copy number separation.
- Median vs mode allele calls:
- HMMSTR reports both the mode and median of the allele distribution. We report both because depending on the distribution of your data, one may be more accurate. As a general rule of thumb, the mode call will be more accurate at higher depths (>30x coverage per allele) while the median will be more accurate and consistent at lower coverage. Usually these metrics will be very similar if not the same, however if there is a significant difference, consider checking the supporting read histogram to make a more informed decision.
- Can I run HMMSTR on PCR-amplified data?
- Yes! Depending on the location of the primers used in the experiment, you may need to adjust HMMSTR parameters to account for short flanking sequence. To account for this, we have used these parameters in our analysis of amplicon data:
hmmstr targets_tsv [Input tsv] [Output prefix] [Infile] --mapq_cutoff 0 --mode sr --k 6 --w 2 --use_full_read --flanking_size 50
- Can I run HMMSTR on whole genome sequencing data?
- HMMSTR is designed for targeted sequencing data and is not optimized for WGS data. However, if you would like to use HMMSTR to genotype specific targets from a WGS dataset we recommend you use
coordinates
in combination with a pre-aligned BAM file as input. This allows for more rigorous target assignment and limits off target genotyping.
- HMMSTR is designed for targeted sequencing data and is not optimized for WGS data. However, if you would like to use HMMSTR to genotype specific targets from a WGS dataset we recommend you use
- How can I call copy number estimates from non-spanning/soft clip reads?
- While a core requirement of the HMMSTR algorithm is detecting unique flanking sequence, you can obtain copy number estimates from soft clipped reads using HMMSTR following our methods in our manuscript. Put briefly, you can arrange your inputs to target one flanking region and allow the second flanking region to end in the expected repeat. Note that this procedure will yield a rough estimate and we do plan to incorporate a more rigorous mode for non-spanning read estimates in future iterations.
- Can I use HMMSTR to recover motif composition?
- HMMSTR does not currently concurrently derive motif composition, however it can be used in conjunction with other motif decomposition softwares and we do so in our in-house processing pipeline. HMMSTR returns the position of the tandem repeat in each read as well as per-read allele assignments which allows for downstream analysis on the repeat sequences.
- I want to make my own visualizations, how can I do this from HMMSTR outputs?
- All of the default visualizations are made from the outputs reported in the *_read_assignemnts.tsv file, you can use this to make your own custom figures