use std::{
fs::File,
io::{self, BufReader, Read},
path::Path,
};
use noodles_bcf as bcf;
use noodles_bgzf as bgzf;
use noodles_csi::BinningIndex;
use noodles_vcf as vcf;
use super::IndexedReader;
use crate::variant::io::{
reader::builder::{detect_compression_method, detect_format},
CompressionMethod, Format,
};
#[derive(Default)]
pub struct Builder {
compression_method: Option<Option<CompressionMethod>>,
format: Option<Format>,
index: Option<Box<dyn BinningIndex>>,
}
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_index<I>(mut self, index: I) -> Self
where
I: BinningIndex + 'static,
{
self.index = Some(Box::new(index));
self
}
pub fn build_from_path<P>(self, src: P) -> io::Result<IndexedReader<bgzf::Reader<File>>>
where
P: AsRef<Path>,
{
let mut reader = File::open(src.as_ref()).map(BufReader::new)?;
let compression_method = match self.compression_method {
Some(compression_method) => compression_method,
None => detect_compression_method(&mut reader)?,
};
let format = match self.format {
Some(format) => format,
None => detect_format(&mut reader, compression_method)?,
};
match (format, compression_method) {
(Format::Vcf, Some(CompressionMethod::Bgzf)) => {
let mut builder = vcf::io::indexed_reader::Builder::default();
if let Some(index) = self.index {
builder = builder.set_index(index);
}
builder.build_from_path(src).map(IndexedReader::Vcf)
}
(Format::Bcf, Some(CompressionMethod::Bgzf)) => {
let mut builder = bcf::io::indexed_reader::Builder::default();
if let Some(index) = self.index {
builder = builder.set_index(index);
}
builder.build_from_path(src).map(IndexedReader::Bcf)
}
(_, None) => Err(io::Error::new(
io::ErrorKind::InvalidData,
"source not bgzip-compressed",
)),
}
}
pub fn build_from_reader<R>(
self,
reader: R,
) -> io::Result<IndexedReader<bgzf::Reader<BufReader<R>>>>
where
R: Read,
{
let mut reader = BufReader::new(reader);
let compression = match self.compression_method {
Some(compression) => compression,
None => detect_compression_method(&mut reader)?,
};
let format = match self.format {
Some(format) => format,
None => detect_format(&mut reader, compression)?,
};
match (format, compression) {
(Format::Vcf, Some(CompressionMethod::Bgzf)) => {
let mut builder = vcf::io::indexed_reader::Builder::default();
if let Some(index) = self.index {
builder = builder.set_index(index);
}
builder.build_from_reader(reader).map(IndexedReader::Vcf)
}
(Format::Bcf, Some(CompressionMethod::Bgzf)) => {
let mut builder = bcf::io::indexed_reader::Builder::default();
if let Some(index) = self.index {
builder = builder.set_index(index);
}
builder.build_from_reader(reader).map(IndexedReader::Bcf)
}
(_, None) => Err(io::Error::new(
io::ErrorKind::InvalidData,
"source not bgzip-compressed",
)),
}
}
}
#[cfg(test)]
mod tests {
use noodles_csi as csi;
use super::*;
#[test]
fn test_build_from_reader() -> io::Result<()> {
let mut writer = bcf::io::Writer::new(Vec::new());
let header = vcf::Header::default();
writer.write_header(&header)?;
writer.try_finish()?;
let data = writer.into_inner().into_inner();
let index = csi::Index::default();
let mut reader = Builder::default()
.set_index(index)
.build_from_reader(&data[..])?;
reader.read_header()?;
Ok(())
}
}