1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
use crate::{Headers, Row};
use csv::WriterBuilder;
use std::fs::{self, File};
use std::io;
use std::path::PathBuf;

pub trait Target {
	/// Useful for initializations
	fn write_headers(&mut self, headers: &Headers) -> Result<(), csv::Error>;
	fn write_row(&mut self, row: &Row) -> Result<(), csv::Error>;
}

pub struct PathTarget {
	path: PathBuf,
	writer: Option<csv::Writer<File>>,
}
impl PathTarget {
	pub fn new<P: Into<PathBuf>>(path: P) -> Self {
		Self {
			path: path.into(),
			writer: None,
		}
	}
}
impl Target for PathTarget {
	fn write_headers(&mut self, headers: &Headers) -> Result<(), csv::Error> {
		if let Some(parent) = self.path.parent() {
			fs::create_dir_all(parent)?;
		}

		self.writer = Some(csv::Writer::from_path(&self.path)?);
		self.write_row(headers.get_row())
	}
	fn write_row(&mut self, row: &Row) -> Result<(), csv::Error> {
		self.writer.as_mut().unwrap().write_record(row)?;
		Ok(())
	}
}

pub struct StdoutTarget {
	writer: Option<csv::Writer<io::Stdout>>,
}
impl StdoutTarget {
	pub fn new() -> Self {
		Self { writer: None }
	}
}
impl Target for StdoutTarget {
	fn write_headers(&mut self, headers: &Headers) -> Result<(), csv::Error> {
		let writer = WriterBuilder::new().from_writer(io::stdout());
		self.writer = Some(writer);
		self.write_row(headers.get_row())?;
		Ok(())
	}
	fn write_row(&mut self, row: &Row) -> Result<(), csv::Error> {
		self.writer.as_mut().unwrap().write_record(row)?;
		Ok(())
	}
}

pub struct StderrTarget {
	writer: Option<csv::Writer<io::Stderr>>,
}
impl StderrTarget {
	pub fn new() -> Self {
		Self { writer: None }
	}
}
impl Target for StderrTarget {
	fn write_headers(&mut self, headers: &Headers) -> Result<(), csv::Error> {
		let writer = WriterBuilder::new().from_writer(io::stderr());
		self.writer = Some(writer);
		self.write_row(headers.get_row())
	}
	fn write_row(&mut self, row: &Row) -> Result<(), csv::Error> {
		self.writer.as_mut().unwrap().write_record(row)?;
		Ok(())
	}
}

pub struct StringWriter<'a> {
	s: &'a mut String,
}
impl<'a> io::Write for StringWriter<'a> {
	fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
		match std::str::from_utf8(buf) {
			Ok(s) => {
				self.s.push_str(s);
				Ok(buf.len())
			}
			Err(e) => Err(io::Error::new(io::ErrorKind::InvalidData, e)),
		}
	}
	fn flush(&mut self) -> io::Result<()> {
		Ok(())
	}
}

pub struct StringTarget<'a> {
	writer: csv::Writer<StringWriter<'a>>,
}
impl<'a> StringTarget<'a> {
	pub fn new(s: &'a mut String) -> Self {
		let writer = WriterBuilder::new().from_writer(StringWriter { s });
		Self { writer }
	}
}
impl<'a> Target for StringTarget<'a> {
	fn write_headers(&mut self, headers: &Headers) -> Result<(), csv::Error> {
		self.write_row(headers.get_row())
	}
	fn write_row(&mut self, row: &Row) -> Result<(), csv::Error> {
		self.writer.write_record(row)?;
		Ok(())
	}
}