use std::{
fs::File,
io::{self, BufWriter, Write},
path::Path,
};
use cram::data_container::BlockContentEncoderMap;
use noodles_bam as bam;
use noodles_cram as cram;
use noodles_fasta as fasta;
use noodles_sam as sam;
use super::Writer;
use crate::alignment::Format;
#[derive(Default)]
pub struct Builder {
format: Option<Format>,
reference_sequence_repository: fasta::Repository,
block_content_encoder_map: BlockContentEncoderMap,
}
impl Builder {
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 fn set_block_content_encoder_map(
mut self,
block_content_encoder_map: BlockContentEncoderMap,
) -> Self {
self.block_content_encoder_map = block_content_encoder_map;
self
}
pub fn build_from_path<P>(mut self, src: P) -> io::Result<Writer>
where
P: AsRef<Path>,
{
let src = src.as_ref();
if self.format.is_none() {
self.format = detect_format_from_path_extension(src);
}
let file = File::create(src).map(BufWriter::new)?;
Ok(self.build_from_writer(file))
}
pub fn build_from_writer<W>(self, writer: W) -> Writer
where
W: Write + 'static,
{
let format = self.format.unwrap_or(Format::Sam);
let inner: Box<dyn sam::AlignmentWriter> = match format {
Format::Sam => Box::new(sam::Writer::new(writer)),
Format::Bam => Box::new(bam::Writer::new(writer)),
Format::Cram => Box::new(
cram::writer::Builder::default()
.set_reference_sequence_repository(self.reference_sequence_repository)
.set_block_content_encoder_map(self.block_content_encoder_map)
.build_with_writer(writer),
),
};
Writer { inner }
}
}
fn detect_format_from_path_extension<P>(path: P) -> Option<Format>
where
P: AsRef<Path>,
{
match path.as_ref().extension().and_then(|ext| ext.to_str()) {
Some("sam") => Some(Format::Sam),
Some("bam") => Some(Format::Bam),
Some("cram") => Some(Format::Cram),
_ => None,
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_detect_format_from_path_extension() {
assert_eq!(
detect_format_from_path_extension("out.sam"),
Some(Format::Sam)
);
assert_eq!(
detect_format_from_path_extension("out.bam"),
Some(Format::Bam)
);
assert_eq!(
detect_format_from_path_extension("out.cram"),
Some(Format::Cram)
);
assert!(detect_format_from_path_extension("out.fa").is_none());
}
}