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
117
118
119
120
121
122
123
124
125
126
|
use crate::models::{Chapter, FileEntry};
use anyhow::{Context, Result};
use relative_path::{RelativePath, RelativePathBuf};
use reqwest::{Client, Url};
use std::{
collections::HashMap,
io::{Read, Write},
path::Path,
};
use tokio::{
fs::{self, File},
io::AsyncWriteExt,
};
use zip::{CompressionMethod, ZipWriter, write::FileOptions};
/// Creates and writes container.xml.
fn write_container_xml_to_zip(
zip: &mut ZipWriter<std::fs::File>,
opf_full_path: &RelativePathBuf,
) -> Result<()> {
// Prepare file contents.
let contents = format!(
r#"<?xml version="1.0" encoding="UTF-8"?>
<container version="1.0" xmlns="urn:oasis:names:tc:opendocument:xmlns:container">
<rootfiles>
<rootfile full-path="{opf_full_path}" media-type="application/oebps-package+xml"/>
</rootfiles>
</container>
"#
);
// Write down the file.
let options: FileOptions<()> =
FileOptions::default().compression_method(CompressionMethod::Deflated);
zip.start_file("META-INF/container.xml", options)?;
zip.write_all(contents.as_bytes())?;
Ok(())
}
pub async fn download_all_files(
client: &Client,
file_entries: &[FileEntry],
dest_root: &Path,
) -> Result<()> {
for entry in file_entries {
let dest_path = entry.full_path.to_path(dest_root);
if let Some(parent_dir) = dest_path.parent() {
fs::create_dir_all(parent_dir).await?;
}
let mut file = File::create(dest_path).await?;
let bytes = client
.get(&entry.url)
.send()
.await?
.error_for_status()?
.bytes()
.await?;
file.write_all(&bytes).await?;
}
Ok(())
}
/// Creates the EPUB archive (creates zip and includes all files in it).
pub fn create_epub_archive(
epub_root: &Path,
output_epub: &Path,
file_entries: &[FileEntry],
chapters: &HashMap<String, Chapter>,
) -> Result<()> {
let out_file = std::fs::File::create(output_epub)?;
let mut zip = ZipWriter::new(out_file);
// Write mimetype to zip first. It must be uncompressed.
let options: FileOptions<()> =
FileOptions::default().compression_method(CompressionMethod::Stored);
zip.start_file("mimetype", options)?;
zip.write_all(b"application/epub+zip")?;
// Find the OPF file entry to reference it in container.xml
let opf_entry = file_entries
.iter()
.find(|f| f.filename_ext == ".opf" && f.media_type == "application/oebps-package+xml")
.context("No OPF file with the correct MIME type was found.")?;
write_container_xml_to_zip(&mut zip, &opf_entry.full_path)?;
// Prepare url path to local path mapping to clean xhtml files from external dependencies.
let url_to_local = file_entries
.iter()
.map(url_path_to_local)
.collect::<Result<HashMap<_, _>>>()?;
// Add the rest of the files according to file_entries.
let options: FileOptions<()> =
FileOptions::default().compression_method(CompressionMethod::Deflated);
for entry in file_entries {
zip.start_file(&entry.full_path, options)?;
let mut src_file = std::fs::File::open(entry.full_path.to_path(epub_root))?;
let mut buffer = Vec::new();
src_file.read_to_end(&mut buffer)?;
if chapters.contains_key(&entry.ourn) {
let mut html = String::from_utf8(buffer)?;
let chapter_dir = entry.full_path.parent().unwrap_or(RelativePath::new(""));
for (url_path, local_path) in &url_to_local {
let rel_path = chapter_dir.relative(local_path);
html = html.replace(url_path, rel_path.as_str());
}
zip.write_all(html.as_bytes())?;
} else {
zip.write_all(&buffer)?;
}
}
zip.finish()?;
Ok(())
}
/// Helper function. Maps FileEntry to (url path, full_path) pair.
fn url_path_to_local(entry: &FileEntry) -> Result<(String, RelativePathBuf)> {
let url = Url::parse(&entry.url).with_context(|| format!("Could not parse: {}", entry.url))?;
let url_path = url.path().to_string();
Ok((url_path, entry.full_path.clone()))
}
|