1 Calculating error rates.

This document is intended to step through the process followed in these estimates of RT error rates. This R package is a companion to the similarly named ‘errrt’ Perl package, which is responsible for preprocessing the raw sequencing data.

2 Introduction

An outline of the experimental process is represented by:

Experimental Design

Experimental Design

The errrt package is responsible for the bottom and right side of the image. It takes the raw fastq data, merges the reads using flash, writes out the index and identifier for all reads identical to the template, writes out the sequences for the ones not identical to the template, aligns them with fasta36 against the template, interprets the fasta36 output, and writes a table of the mismatches, insertions, deletions from it.

This package, creatively named ‘errRt’ takes a sample sheet describing the samples, reads the tables of identical reads/mismatches, and creates matrices of mutation rates for various classes of mutations.

3 Installation

The perl package may be installed via:

The R package may be installed via:

4 Preprocessing Steps

I split up the preprocessing into discrete steps so that I could check on each phase and try to ensure correctness. The errrt Perl package was written so that pretty much everything may be set via command line parameters, but all the defaults were written to suite this data set. Thus I have a pretty simple bash for loop which handled the work of preprocessing the data.

The following fragment of bash was run inside my directory preprocessing/, which contained 3 subdirectories: s1/, s2/, and s3/ which contain the raw reads for the 3 samples. The excel spreadsheet in inst/extdata/all_samples.xlsx describes them, but s1 is the control, s2 has low magnesium, and s3 has high.

As the above shell fragment states, the final outputs of errrt are step4.txt.xz and step2_identical_reads.txt.xz. These filenames are therefore recorded in the sample sheet in the columns ‘ident_table’ and ‘mutation_table’ for each sample.

I decided for the moment at least to copy these files into inst/extdata and modify the sample sheet in inst/extdata to refer to them.

5 Processing the data

I wrapped the various steps performed by errRt into a primary main function, create_matrices(). The following is the invocation I used for this data, along with some explanation of what is happening.

5.1 Getting Started

## [1] "/tmp/RtmpPGuTJL/temp_libpath9dc2c624a249c/errRt/extdata/all_samples.xlsx"

Here is the resulting experimental design. The create_matrices() function will use this to read the tables of identical reads and mutated reads.

sampleid condition batch identtable mutationtable
s1 s1 control b201910 s1/step2_identical_reads.txt.xz s1/step4.txt.xz
s2 s2 low b201910 s2/step2_identical_reads.txt.xz s2/step4.txt.xz
s3 s3 high b201910 s3/step2_identical_reads.txt.xz s3/step4.txt.xz

5.2 Create matrices

The create_matrices() function has a few options to adjust the stringency of the various ways to define an error. It functions to count the various error types with quantify_parsed(), gather them into matrices by sample along with the perceived errors by the sequencer, and calculate aggregate error rates for the various classes of error. In other words, it does pretty much everything in one function call.

## Starting sample: 1.
## Reading the file containing mutations: s1/step4.txt.xz
## Reading the file containing the identical reads: s1/step2_identical_reads.txt.xz
## Removing any differences before position: 24.
## Before pruning, there are: 1156535 reads.
## After position pruning, there are: 1037310 reads: 90%.
## Removing any reads with 'N' as the hit.
## Before N pruning, there are: 1037310 reads.
## After N pruning, there are: 1021066 reads: 98%.
## Gathering information about the indexes observed, this is slow.
## Before read pruning, there are: 1742743 indexes.
## After read pruning, there are: 838158 indexes.
## Removing indexes with fewer than 3 indexes.
## Before index pruning, there are: 1021066 changed reads.
## Before index pruning, there are: 4681501 identical reads.
## After index pruning, there are: 532571 changed reads: 52%.
## After index pruning, there are: 3663814 identical reads: 78%.
## Gathering identical, mutant, and sequencer reads/indexes.
## Counting by direction.
## Counting by string.
## Subsetting based on mutations with at least 3 indexes.
## Counting by reference position.
## Counting by identity string.
## Counting by reference nucleotide.
## Counting by product nucleotide.
## Counting by mutation type.
## Counting by transitions/transversion.
## Counting strong/weak.
## Counting insertions by position.
## Counting insertions by nucleotide
## Counting deletions by position.
## Counting deletions by nucleotide.
## Starting sample: 2.
## Reading the file containing mutations: s2/step4.txt.xz
## Reading the file containing the identical reads: s2/step2_identical_reads.txt.xz
## Removing any differences before position: 24.
## Before pruning, there are: 3421203 reads.
## After position pruning, there are: 1758479 reads: 51%.
## Removing any reads with 'N' as the hit.
## Before N pruning, there are: 1758479 reads.
## After N pruning, there are: 1732605 reads: 99%.
## Gathering information about the indexes observed, this is slow.
## Before read pruning, there are: 1263563 indexes.
## After read pruning, there are: 694479 indexes.
## Removing indexes with fewer than 3 indexes.
## Before index pruning, there are: 1732605 changed reads.
## Before index pruning, there are: 5230976 identical reads.
## After index pruning, there are: 874781 changed reads: 50%.
## After index pruning, there are: 4834367 identical reads: 92%.
## Gathering identical, mutant, and sequencer reads/indexes.
## Counting by direction.
## Counting by string.
## Subsetting based on mutations with at least 3 indexes.
## Counting by reference position.
## Counting by identity string.
## Counting by reference nucleotide.
## Counting by product nucleotide.
## Counting by mutation type.
## Counting by transitions/transversion.
## Counting strong/weak.
## Counting insertions by position.
## Counting insertions by nucleotide
## Counting deletions by position.
## Counting deletions by nucleotide.
## Starting sample: 3.
## Reading the file containing mutations: s3/step4.txt.xz
## Reading the file containing the identical reads: s3/step2_identical_reads.txt.xz
## Removing any differences before position: 24.
## Before pruning, there are: 4309681 reads.
## After position pruning, there are: 1564155 reads: 36%.
## Removing any reads with 'N' as the hit.
## Before N pruning, there are: 1564155 reads.
## After N pruning, there are: 1532016 reads: 98%.
## Gathering information about the indexes observed, this is slow.
## Before read pruning, there are: 887982 indexes.
## After read pruning, there are: 465046 indexes.
## Removing indexes with fewer than 3 indexes.
## Before index pruning, there are: 1532016 changed reads.
## Before index pruning, there are: 3583390 identical reads.
## After index pruning, there are: 783358 changed reads: 51%.
## After index pruning, there are: 3332425 identical reads: 93%.
## Gathering identical, mutant, and sequencer reads/indexes.
## Counting by direction.
## Counting by string.
## Subsetting based on mutations with at least 3 indexes.
## Counting by reference position.
## Counting by identity string.
## Counting by reference nucleotide.
## Counting by product nucleotide.
## Counting by mutation type.
## Counting by transitions/transversion.
## Counting strong/weak.
## Counting insertions by position.
## Counting insertions by nucleotide
## Counting deletions by position.
## Counting deletions by nucleotide.
## Working on miss_reads_by_position.
## Working on miss_indexes_by_position.
## Working on miss_sequencer_by_position.
## Working on miss_reads_by_string.
## Warning in order(as.numeric(rownames(matrices[[t]]))): NAs introduced by
## coercion
## Working on miss_indexes_by_string.
## Warning in order(as.numeric(rownames(matrices[[t]]))): NAs introduced by
## coercion
## Working on miss_sequencer_by_string.
## Warning in order(as.numeric(rownames(matrices[[t]]))): NAs introduced by
## coercion
## Working on miss_reads_by_ref_nt.
## Working on miss_indexes_by_ref_nt.
## Working on miss_sequencer_by_ref_nt.
## Working on miss_reads_by_hit_nt.
## Working on miss_indexes_by_hit_nt.
## Working on miss_sequencer_by_hit_nt.
## Working on miss_reads_by_type.
## Working on miss_indexes_by_type.
## Working on miss_sequencer_by_type.
## Working on miss_reads_by_trans.
## Working on miss_indexes_by_trans.
## Working on miss_sequencer_by_trans.
## Working on miss_reads_by_strength.
## Working on miss_indexes_by_strength.
## Working on miss_sequencer_by_strength.
## Working on insert_reads_by_position.
## Working on insert_indexes_by_position.
## Working on insert_sequencer_by_position.
## Working on insert_reads_by_nt.
## Working on insert_indexes_by_nt.
## Working on insert_sequencer_by_nt.
## Working on delete_reads_by_position.
## Working on delete_indexes_by_position.
## Working on delete_sequencer_by_position.
## Working on delete_reads_by_nt.
## Working on delete_indexes_by_nt.
## Working on delete_sequencer_by_nt.
## Skipping table: delete_reads_by_position
## Skipping table: delete_indexes_by_position
## Skipping table: delete_sequencer_by_position
## Skipping table: delete_reads_by_nt
## Skipping table: delete_indexes_by_nt
## Skipping table: delete_sequencer_by_nt

The options are explained in the help documentation, but here is an overview of the ones which might not be self-evident.

  • min_reads: How many reads/index are required for a given index to be considered usable? This is the first toggle of stringency and is affected primarily by the sequencing depth.
  • min_indexes: How many separate indexes are expected to agree on a given mutation in order for it to be considered real? This toggle of stringency is primarily affected by the success in getting a large number of different RT primers to bind and extend during the early stages of creating the sequencing libraries.
  • min_sequencer: In order for a given mutation to be deemed ‘sequencer-specific’, 1 and only 1 read out of ‘n’ for a given index must have the mutation. This toggles ‘n’, higher numbers are thus more stringent and require greater sequencing depth. If this is set too low, a pool of false positives will result; the default of 10 is completely arbitrary.
  • min_position: In the actual experiment, it appears that the primer used for the experimental samples may have had an error, or it was differently cut out of the PAGE purification gel, or something else happened which resulted in a tremendous percentage of the reads presenting themselves as either a deletion at ~ position 20, or a A to C mutation at position 22 or somesuch. This parameter just ignores any mutations occuring before this position. We observed that this strange trend stopped at position 22, and so set it to 24 to be safe.
  • prune_n: When true, this simply drops entries in the mutation table which have Ns.

create_matrices() works in 3 main stages:

  1. For each sample in the metadata, invoke quantify_parsed(). This function is responsible for reading the tables and collating the various error types. This is the most interesting portion of code and most likely to have problems.
  2. Collect the various tables from step 1 and merge them into 1 matrix per table where the rows are mutation types and columns are samples.
  3. Calculate error rates by read/index and write new versions of the matrices from step 2 accordingly.

While create_matrices() is running, it should print some information about what it is doing and give some idea about how much data is lost due to the stringency parameters. Its result is a list:

  • samples: The result from quantify_parsed() for each sample in the metadata.
  • reads_per_sample: The sum of the reads observed in the mutation data and the identity data, e.g. how many reads survived.
  • indexes_per_sample: The sum of the indexes observed. This is our primary normalization factor, but along with reads_per_sample should help define a healthy vs. problematic sample.
  • matrices: The ‘raw’ matrices for each table.
  • matrices_by_counts: The matrices dividied by the indexes.
  • normalized: The ‘raw’ matrices as cpm, thus normalized by library sizes.
  • normalized_by_counts: The matrices as cpm and dividied by indexes.

Thus, as one might imagine, there are a lot of matrices returned, here are some examples of the data after normalization, in each case I will first print the data for the RT mutations followed by the sequencer:

  1. Mismatches by position as a heatmap: This is quite curious, as we see apparently fewer variants at the end of the sequence.
## Warning in gplots::heatmap.2(error_data[["normalized_by_counts"]]
## [["miss_indexes_by_position"]], : Discrepancy: Rowv is FALSE, while dendrogram
## is `both'. Omitting row dendogram.

## Warning in gplots::heatmap.2(error_data[["normalized_by_counts"]]
## [["miss_sequencer_by_position"]], : Discrepancy: Rowv is FALSE, while dendrogram
## is `both'. Omitting row dendogram.

  1. Mismatches by reference nucleotide
s1 s2 s3
A 0.0965 0.2340 0.5623
C 0.6078 0.8293 0.4911
G 0.8937 0.3998 0.3085
T 0.0971 0.1681 0.2422
s1 s2 s3
A 0.3294 0.3464 0.5231
C 0.3180 0.5133 0.2713
G 0.3688 0.2011 0.2530
T 0.3956 0.5071 0.7605
  1. Mismatches by mutated nucleotide
s1 s2 s3
A 0.4794 0.7054 0.6678
C 0.1254 0.1721 0.1012
G 0.1745 0.2213 0.4140
T 0.5130 0.3522 0.6819
s1 s2 s3
A 0.2307 0.2967 0.4339
C 0.4693 0.6286 0.3616
G 0.6785 0.4078 0.4838
T 0.1971 0.2302 0.3422
  1. Mismatches by type: A to C, G to A etc etc
s1 s2 s3
A_C 0.0553 0.0404 0.0433
A_G 0.0376 0.1609 0.2962
A_T 0.0181 0.0362 0.0419
C_A 0.3978 0.2640 0.1602
C_G 0.0176 0.0416 0.0557
C_T 0.1644 0.2914 0.5133
G_A 0.0720 0.2747 0.1711
G_C 0.0146 0.0182 0.0171
G_T 0.7420 0.1984 0.1267
T_A 0.0095 0.0458 0.0392
T_C 0.0441 0.0483 0.0528
T_G 0.0921 0.0964 0.0927
s1 s2 s3
A_C 0.2403 0.1817 0.1850
A_G 0.0803 0.0892 0.0884
A_T 0.0407 0.0565 0.0576
C_A 0.1290 0.1515 0.1462
C_G 0.0732 0.0728 0.0661
C_T 0.1331 0.1315 0.1267
G_A 0.0560 0.0518 0.0526
G_C 0.0580 0.0759 0.0838
G_T 0.1813 0.1558 0.1579
T_A 0.0457 0.0425 0.0419
T_C 0.1214 0.1257 0.1293
T_G 0.4494 0.4930 0.4918
  1. Mismatches by transition/transversion
s1 s2 s3
transition 0.2309 1.0989 0.8993
transversion 1.1613 0.5834 0.8074
s1 s2 s3
transition 0.297 0.5457 0.366
transversion 1.081 0.8903 1.604
  1. Mismatches by strong/weak
s1 s2 s3
strong_strong 0.0267 0.0598 0.1087
strong_weak 1.1740 1.4607 0.6864
weak_strong 0.3137 0.2672 0.4634
weak_weak 0.0195 0.0795 0.1125
s1 s2 s3
strong_strong 0.1087 0.1487 0.2238
strong_weak 0.4338 0.6537 0.3568
weak_strong 1.1836 0.6333 0.7703
weak_weak 0.0683 0.0892 0.1332
  1. Insertions by position
s2 s3
30 0.0332 0.0226
46 0.0000 0.0076
48 0.2105 0.2173
51 0.4135 0.6765
52 0.0000 0.0339
55 0.0000 0.0302
72 0.0000 0.1101
81 0.0000 0.0151
91 0.0886 0.2032
92 0.3308 0.0661
93 0.0000 0.0169
97 0.6616 0.1568
100 0.0332 0.0000
123 0.0000 0.0454
142 0.0000 0.0198
144 0.0000 0.0170
148 0.0000 0.0113
151 0.0662 0.0000
154 0.0886 0.0000
s2 s3
29 0.1440 0.1129
34 0.0000 0.1265
35 0.1080 0.0000
42 0.0000 0.1687
46 0.0000 0.0847
48 0.2688 0.2108
51 0.1080 0.1412
55 0.0000 0.1265
66 0.1440 0.0000
92 0.8601 0.5903
97 0.0000 0.0847
149 0.0000 0.1687
151 0.0000 0.0847
156 0.2688 0.0000
  1. Insertions by nt
s2 s3
A 0.2769 0.7219
C 0.3804 0.1975
G 0.3434 0.1682
T 0.8436 0.6237
s2 s3
A 0.108 0.2259
C 1.236 1.0119
G 0.504 0.4517
T 0.000 0.1265
  1. Deletions by position

There was insufficient data to quantify these.

---
title: "Counting RT mutations from illumina sequencing data."
author: "atb abelew@gmail.com"
date: "`r Sys.Date()`"
output:
  html_document:
    code_download: true
    code_folding: show
    fig_caption: true
    fig_height: 7
    fig_width: 7
    highlight: tango
    keep_md: false
    mode: selfcontained
    number_sections: true
    self_contained: true
    theme: readable
    toc: true
    toc_float:
      collapsed: false
      smooth_scroll: false
  rmdformats::readthedown:
    code_download: true
    code_folding: show
    df_print: paged
    fig_caption: true
    fig_height: 7
    fig_width: 7
    highlight: tango
    width: 300
    keep_md: false
    mode: selfcontained
    toc_float: true
  BiocStyle::html_document:
    code_download: true
    code_folding: show
    fig_caption: true
    fig_height: 7
    fig_width: 7
    highlight: tango
    keep_md: false
    mode: selfcontained
    toc_float: true
vignette: >
  %\VignetteIndexEntry{error_quantification}
  %\VignetteEngine{knitr::rmarkdown}
  \usepackage[utf8]{inputenc}
---

<style type="text/css">
body, td {
  font-size: 16px;
}
code.r{
  font-size: 16px;
}
pre {
 font-size: 16px
}
</style>

```{r options, include=FALSE}
library("errRt")
basedir <- system.file("extdata", package="errRt")
knitr::opts_knit$set(width=120,
                     root.dir=basedir,
                     progress=TRUE,
                     verbose=TRUE,
                     echo=TRUE)
knitr::opts_chunk$set(error=TRUE,
                      dpi=96)
old_options <- options(digits=4,
                       stringsAsFactors=FALSE,
                       knitr.duplicate.label="allow")
ggplot2::theme_set(ggplot2::theme_bw(base_size=10))
rundate <- format(Sys.Date(), format="%Y%m%d")
##tmp <- sm(loadme(filename=paste0(gsub(pattern="\\.Rmd", replace="", x=previous_file), "-v", ver, ".rda.xz")))
##rmd_file <- "03_expression_infection_20180822.Rmd"
```

# Calculating error rates.

This document is intended to step through the process followed in these
estimates of RT error rates.  This R package is a companion to the similarly
named 'errrt' Perl package, which is responsible for preprocessing the raw
sequencing data.

# Introduction

An outline of the experimental process is represented by:

![Experimental Design](experimental_design.svg)

The errrt package is responsible for the bottom and right side of the image.  It takes the
raw fastq data, merges the reads using flash, writes out the index and
identifier for all reads identical to the template, writes out the sequences for
the ones not identical to the template, aligns them with fasta36 against the
template, interprets the fasta36 output, and writes a table of the mismatches,
insertions, deletions from it.

This package, creatively named 'errRt' takes a sample sheet describing the
samples, reads the tables of identical reads/mismatches, and creates matrices
of mutation rates for various classes of mutations.

# Installation

The perl package may be installed via:

```{bash install_errrt, eval=FALSE}
git clone https://github.com/abelew/errrt
cd errrt
perl Build.PL
perl Build install
```

The R package may be installed via:

```{r installation, eval=FALSE}
devtools::install_github("abelew/errRt")
```

# Preprocessing Steps

I split up the preprocessing into discrete steps so that I could check on each
phase and try to ensure correctness.  The errrt Perl package was written so that
pretty much everything may be set via command line parameters, but all the
defaults were written to suite this data set.  Thus I have a pretty simple bash
for loop which handled the work of preprocessing the data.

The following fragment of bash was run inside my directory *preprocessing/*,
which contained 3 subdirectories: *s1/*, *s2/*, and *s3/* which contain the
raw reads for the 3 samples.  The excel spreadsheet in
*inst/extdata/all_samples.xlsx* describes them, but s1 is the control, s2 has
low magnesium, and s3 has high.

```{bash preprocessing, eval=FALSE}
#!/usr/bin/env bash

## The default options should take care of most of what these require.
start=$(pwd)
for sample in $(/bin/ls -d s*); do
    cd ${sample}
    ## Merge the forward and reverse reads.
    ../../errrt/script/merge_reads.pl --read1 *_1.fastq.gz --read2 *_2.fastq.gz
    ## Extract the reads with the template sequence into a fasta file.
    ## This assumes input named 'step1.extendedFrags.fastq.xz' and output named 'step2.fasta'
    ## This output is not compressed because it is easier to invoke fasta36 on uncompressed data.
    ## Thus it defaults to step2.fasta as output.
    ../../errrt/script/extract_fasta.pl
    ## Run fasta against the template sequence.
    ## This assumes step2.fasta as input and provides step3.txt.xz as output.
    ../../errrt/script/run_fasta.pl
    ## Boil the fasta output into something we can read.
    ## This assumes step3.txt.xz as input and step4.txt.xz as output.
    ../../errrt/script/parse_fasta.pl
    cd ${start}
done
```

As the above shell fragment states, the final outputs of errrt are *step4.txt.xz*
and *step2_identical_reads.txt.xz*.  These filenames are therefore recorded in
the sample sheet in the columns 'ident_table' and 'mutation_table' for each
sample.

I decided for the moment at least to copy these files into inst/extdata and
modify the sample sheet in inst/extdata to refer to them.

# Processing the data

I wrapped the various steps performed by errRt into a primary main function,
*create_matrices()*.  The following is the invocation I used for this data,
along with some explanation of what is happening.

## Getting Started

```{r getting_started}
library(errRt)
basedir <- system.file("extdata", package="errRt")
metadata_file <- file.path(basedir, "all_samples.xlsx")
metadata_file
setwd(basedir)
meta <- hpgltools::extract_metadata(metadata_file)
```

Here is the resulting experimental design.  The *create_matrices()* function
will use this to read the tables of identical reads and mutated reads.

```{r metadata, results='asis'}
knitr::kable(meta)
```

## Create matrices

The *create_matrices()* function has a few options to adjust the
stringency of the various ways to define an error.  It functions to count the
various error types with *quantify_parsed()*, gather them into matrices by
sample along with the perceived errors by the sequencer, and calculate aggregate
error rates for the various classes of error.  In other words, it does pretty
much everything in one function call.

```{r create_matrices}
error_data <- create_matrices(sample_sheet=metadata_file, ident_column="identtable",
                              mut_column="mutationtable", min_reads=3, min_indexes=3,
                              min_sequencer=10, min_position=24, prune_n=TRUE,
                              verbose=TRUE)
```

The options are explained in the help documentation, but here is an overview of
the ones which might not be self-evident.

*  min_reads:  How many reads/index are required for a given index to be
   considered usable?  This is the first toggle of stringency and is affected
   primarily by the sequencing depth.
*  min_indexes:  How many separate indexes are expected to agree on a given
   mutation in order for it to be considered real?  This toggle of stringency is
   primarily affected by the success in getting a large number of different RT
   primers to bind and extend during the early stages of creating the sequencing
   libraries.
*  min_sequencer:  In order for a given mutation to be deemed
   'sequencer-specific', 1 and only 1 read out of 'n' for a given index must have
   the mutation.  This toggles 'n', higher numbers are thus more stringent and
   require greater sequencing depth.  If this is set too low, a pool of
   false positives will result; the default of 10 is completely arbitrary.
*  min_position:  In the actual experiment, it appears that the primer used for
   the experimental samples may have had an error, or it was differently cut out
   of the PAGE purification gel, or something else happened which resulted in a
   tremendous percentage of the reads presenting themselves as either a deletion
   at ~ position 20, or a A to C mutation at position 22 or somesuch.  This
   parameter just ignores any mutations occuring before this position.  We
   observed that this strange trend stopped at position 22, and so set it to 24
   to be safe.
*  prune_n:  When true, this simply drops entries in the mutation table which
   have Ns.

*create_matrices()* works in 3 main stages:

1.  For each sample in the metadata, invoke *quantify_parsed()*.  This function
    is responsible for reading the tables and collating the various error
    types.  This is the most interesting portion of code and most likely to have
    problems.
2.  Collect the various tables from step 1 and merge them into 1 matrix per
    table where the rows are mutation types and columns are samples.
3.  Calculate error rates by read/index and write new versions of the matrices
    from step 2 accordingly.

While *create_matrices()* is running, it should print some information
about what it is doing and give some idea about how much data is lost due to the
stringency parameters.  Its result is a list:

*  samples: The result from *quantify_parsed()* for each sample in the metadata.
*  reads_per_sample: The sum of the reads observed in the mutation data and the
   identity data, e.g. how many reads survived.
*  indexes_per_sample: The sum of the indexes observed.  This is our primary
   normalization factor, but along with reads_per_sample should help define a
   healthy vs. problematic sample.
*  matrices: The 'raw' matrices for each table.
*  matrices_by_counts: The matrices dividied by the indexes.
*  normalized: The 'raw' matrices as cpm, thus normalized by library sizes.
*  normalized_by_counts: The matrices as cpm and dividied by indexes.

Thus, as one might imagine, there are a _lot_ of matrices returned, here are
some examples of the data after normalization, in each case I will first print
the data for the RT mutations followed by the sequencer:

1.  Mismatches by position as a heatmap: This is quite curious, as we see
    apparently fewer variants at the end of the sequence.

```{r mismatches_by_position}
gplots::heatmap.2(error_data[["normalized_by_counts"]][["miss_indexes_by_position"]],
                  trace="none",
                  Rowv=FALSE)
gplots::heatmap.2(error_data[["normalized_by_counts"]][["miss_sequencer_by_position"]],
                  trace="none",
                  Rowv=FALSE)
```

2.  Mismatches by reference nucleotide

```{r reference_nucleotides, results='asis'}
knitr::kable(error_data[["normalized_by_counts"]][["miss_indexes_by_ref_nt"]])
knitr::kable(error_data[["normalized_by_counts"]][["miss_sequencer_by_ref_nt"]])
```

3.  Mismatches by mutated nucleotide

```{r mutated_nucleotides, results='asis'}
knitr::kable(error_data[["normalized_by_counts"]][["miss_indexes_by_hit_nt"]])
knitr::kable(error_data[["normalized_by_counts"]][["miss_sequencer_by_hit_nt"]])
```

4.  Mismatches by type: A to C, G to A etc etc

```{r type_nucleotides, results='asis'}
knitr::kable(error_data[["normalized_by_counts"]][["miss_indexes_by_type"]])
knitr::kable(error_data[["normalized_by_counts"]][["miss_sequencer_by_type"]])
```

5.  Mismatches by transition/transversion

```{r trans_nucleotides, results='asis'}
knitr::kable(error_data[["normalized_by_counts"]][["miss_indexes_by_trans"]])
knitr::kable(error_data[["normalized_by_counts"]][["miss_sequencer_by_trans"]])
```

6.  Mismatches by strong/weak

```{r strength_nucleotides, results='asis'}
knitr::kable(error_data[["normalized_by_counts"]][["miss_indexes_by_strength"]])
knitr::kable(error_data[["normalized_by_counts"]][["miss_sequencer_by_strength"]])
```

7.  Insertions by position

```{r insert_position, results='asis'}
knitr::kable(error_data[["normalized_by_counts"]][["insert_indexes_by_position"]])
knitr::kable(error_data[["normalized_by_counts"]][["insert_sequencer_by_position"]])
```

8.  Insertions by nt

```{r insert_nt, results='asis'}
knitr::kable(error_data[["normalized_by_counts"]][["insert_indexes_by_nt"]])
knitr::kable(error_data[["normalized_by_counts"]][["insert_sequencer_by_nt"]])
```

9.  Deletions by position

There was insufficient data to quantify these.
