extern crate tempdir;
extern crate walkdir;
extern crate zip;
extern crate quick_xml;
use self::quick_xml::events::attributes::Attribute;
use self::quick_xml::events::{Event, BytesEnd, BytesStart, BytesText};
use self::quick_xml::writer::Writer;
use self::zip::write::FileOptions;
use self::walkdir::WalkDir;
use self::tempdir::TempDir;
use std::borrow::Cow;
use std::fs::File;
use std::fs;
use std::io;
use std::io::{Write,Read,Cursor};
use std::path::Path;
use std::string::FromUtf8Error;
pub fn write_to_file(path: &Path, dir: &TempDir) -> Result<(), io::Error>
{
let file = File::create(&path)?;
let mut zip = zip::ZipWriter::new(file);
let options = FileOptions::default()
.compression_method(zip::CompressionMethod::Deflated)
.unix_permissions(0o644);
let walkdir = WalkDir::new(dir.path());
let it = walkdir.into_iter();
for dent in it.filter_map(|e| e.ok()) {
let path = dent.path();
let name = path.strip_prefix(Path::new(dir.path()))
.unwrap()
.to_str()
.unwrap();
if path.is_file() {
try!(zip.start_file(name, options));
let mut f = File::open(path)?;
let mut buffer = Vec::new();
f.read_to_end(&mut buffer)?;
zip.write_all(&*buffer)?;
}
}
zip.finish()?;
Ok(())
}
pub fn unzip(zip_file: &File, dir: &TempDir) -> Result<(), zip::result::ZipError>
{
let mut zip = zip::ZipArchive::new(zip_file)?;
for i in 0..zip.len() {
let mut file = zip.by_index(i)?;
let path = dir.path().join(file.name());
if let Some(parent) = path.parent() {
fs::create_dir_all(parent)?
}
if (&*file.name()).ends_with("/") {
fs::create_dir_all(path)?
} else {
let mut archived_file = File::create(path)?;
let _ = io::copy(&mut file, &mut archived_file);
}
}
Ok(())
}
pub fn make_static_file(temp_dir: &TempDir, path: &str, data: &str, dir: Option<&str>) -> Result<(), io::Error>
{
match dir {
Some(dir) => {
let dir_path = temp_dir.path().join(dir);
fs::create_dir_all(dir_path)?;
},
None => {}
}
let file_path = temp_dir.path().join(path);
let mut f = File::create(file_path)?;
f.write_all(data.as_bytes())?;
f.sync_all()?;
Ok(())
}
pub fn write_start_tag<'a, S>(writer: &mut Writer<Cursor<Vec<u8>>>, tag_name: S, attributes: Vec<(&str, &str)>, empty_flag: bool)
where S: Into<Cow<'a, str>>
{
let tag_name = tag_name.into();
let mut elem = BytesStart::owned(tag_name.as_bytes().to_vec(), tag_name.len());
for attribute in &attributes {
elem.push_attribute((attribute.0, attribute.1));
}
if empty_flag {
let _ = writer.write_event(Event::Empty(elem));
} else {
let _ = writer.write_event(Event::Start(elem));
}
}
pub fn write_end_tag<'a, S>(writer: &mut Writer<Cursor<Vec<u8>>>, tag_name: S)
where S: Into<Cow<'a, str>>
{
let _ = writer.write_event(Event::End(BytesEnd::borrowed(tag_name.into().as_bytes())));
}
pub fn write_text_node<'a, S>(writer: &mut Writer<Cursor<Vec<u8>>>, data: S)
where S: Into<Cow<'a, str>>
{
let _ = writer.write_event(Event::Text(BytesText::borrowed(data.into().as_bytes())));
}
pub fn make_file_from_writer(path: &str, temp_dir: &TempDir, writer: Writer<Cursor<Vec<u8>>>, dir: Option<&str>) -> Result<(), io::Error> {
match dir {
Some(dir) => {
let dir_path = temp_dir.path().join(dir);
fs::create_dir_all(dir_path)?;
},
None => {}
}
let file_path = temp_dir.path().join(path);
let mut f = File::create(file_path)?;
f.write_all(writer.into_inner().get_ref())?;
f.sync_all()?;
Ok(())
}
pub fn get_attribute_value(attr: &Attribute) -> Result<String, FromUtf8Error> {
let value = (&attr.value).clone().into_owned();
String::from_utf8(value)
}