use std::path::Path;
use noodles_bam as bam;
use noodles_bgzf as bgzf;
use noodles_cram as cram;
use noodles_fasta as fasta;
use noodles_sam as sam;
use tokio::{
fs::File,
io::{self, AsyncBufRead, AsyncBufReadExt, AsyncRead, BufReader},
};
use super::Reader;
use crate::alignment::io::{CompressionMethod, Format};
#[derive(Default)]
pub struct Builder {
compression_method: Option<Option<CompressionMethod>>,
format: Option<Format>,
reference_sequence_repository: fasta::Repository,
}
impl Builder {
pub fn set_compression_method(mut self, compression_method: Option<CompressionMethod>) -> Self {
self.compression_method = Some(compression_method);
self
}
pub fn set_format(mut self, format: Format) -> Self {
self.format = Some(format);
self
}
pub fn set_reference_sequence_repository(
mut self,
reference_sequence_repository: fasta::Repository,
) -> Self {
self.reference_sequence_repository = reference_sequence_repository;
self
}
pub async fn build_from_path<P>(
self,
src: P,
) -> io::Result<Reader<Box<dyn AsyncBufRead + Unpin>>>
where
P: AsRef<Path>,
{
let file = File::open(src).await?;
self.build_from_reader(file).await
}
pub async fn build_from_reader<R>(
self,
reader: R,
) -> io::Result<Reader<Box<dyn AsyncBufRead + Unpin>>>
where
R: AsyncRead + Unpin + 'static,
{
use crate::alignment::io::reader::builder::{detect_compression_method, detect_format};
let mut reader = BufReader::new(reader);
let compression_method = match self.compression_method {
Some(compression_method) => compression_method,
None => {
let mut src = reader.fill_buf().await?;
detect_compression_method(&mut src)?
}
};
let format = match self.format {
Some(format) => format,
None => {
let mut src = reader.fill_buf().await?;
detect_format(&mut src, compression_method)?
}
};
let reader: Box<dyn AsyncBufRead + Unpin> = match (format, compression_method) {
(Format::Sam, None) => Box::new(reader),
(Format::Sam, Some(CompressionMethod::Bgzf)) => {
Box::new(bgzf::AsyncReader::new(reader))
}
(Format::Bam, None) => Box::new(reader),
(Format::Bam, Some(CompressionMethod::Bgzf)) => {
Box::new(bgzf::AsyncReader::new(reader))
}
(Format::Cram, None) => {
let inner: Box<dyn AsyncBufRead + Unpin> = Box::new(reader);
let inner = cram::r#async::io::reader::Builder::default()
.set_reference_sequence_repository(self.reference_sequence_repository)
.build_from_reader(inner);
return Ok(Reader::Cram(inner));
}
(Format::Cram, Some(CompressionMethod::Bgzf)) => {
return Err(io::Error::new(
io::ErrorKind::InvalidData,
"invalid format compression method: CRAM cannot be bgzip-compressed",
));
}
};
let reader: Reader<Box<dyn AsyncBufRead + Unpin>> = match format {
Format::Sam => Reader::Sam(sam::r#async::io::Reader::new(reader)),
Format::Bam => Reader::Bam(bam::r#async::io::Reader::from(reader)),
Format::Cram => unreachable!(), };
Ok(reader)
}
}