Merge branch 'subtree'
[fanfix.git] / src / be / nikiroo / fanfix / supported / Epub.java
CommitLineData
08fe2e33
NR
1package be.nikiroo.fanfix.supported;
2
08fe2e33 3import java.io.File;
08fe2e33
NR
4import java.io.IOException;
5import java.io.InputStream;
7445f856 6import java.net.URISyntaxException;
08fe2e33 7import java.net.URL;
778d8d85 8import java.net.URLDecoder;
68686a37 9import java.util.ArrayList;
15da4d0a
NR
10import java.util.Arrays;
11import java.util.Collections;
08fe2e33
NR
12import java.util.zip.ZipEntry;
13import java.util.zip.ZipInputStream;
14
7445f856
NR
15import org.jsoup.nodes.Document;
16
08fe2e33 17import be.nikiroo.fanfix.Instance;
68686a37 18import be.nikiroo.fanfix.data.MetaData;
08fe2e33 19import be.nikiroo.utils.IOUtils;
16a81ef7 20import be.nikiroo.utils.Image;
b7afbe42 21import be.nikiroo.utils.StringUtils;
8d59ce07 22import be.nikiroo.utils.streams.MarkableFileInputStream;
08fe2e33
NR
23
24/**
25 * Support class for EPUB files created with this program (as we need some
26 * metadata available in those we create).
27 *
28 * @author niki
29 */
68686a37 30class Epub extends InfoText {
7445f856 31 private MetaData meta;
2aac79c7 32 private File tmpDir;
b7afbe42 33 private String desc;
08fe2e33 34
68686a37
NR
35 private URL fakeSource;
36 private InputStream fakeIn;
08fe2e33 37
7445f856 38 public File getSourceFileOriginal() {
298d405a 39 return super.getSourceFile();
08fe2e33
NR
40 }
41
42 @Override
7445f856
NR
43 protected File getSourceFile() {
44 try {
45 return new File(fakeSource.toURI());
46 } catch (URISyntaxException e) {
076caecc
NR
47 Instance.getInstance().getTraceHandler().error(new IOException(
48 "Cannot get the source file from the info-text URL", e));
08fe2e33
NR
49 }
50
51 return null;
52 }
53
54 @Override
7445f856 55 protected InputStream getInput() {
298d405a
NR
56 if (fakeIn != null) {
57 try {
58 fakeIn.reset();
59 } catch (IOException e) {
076caecc
NR
60 Instance.getInstance().getTraceHandler().error(new IOException(
61 "Cannot reset the Epub Text stream", e));
298d405a
NR
62 }
63
64 return fakeIn;
65 }
66
67 return null;
08fe2e33
NR
68 }
69
70 @Override
7445f856
NR
71 protected boolean supports(URL url) {
72 return url.getPath().toLowerCase().endsWith(".epub");
73 }
08fe2e33 74
7445f856
NR
75 @Override
76 protected MetaData getMeta() throws IOException {
77 return meta;
08fe2e33
NR
78 }
79
80 @Override
7445f856
NR
81 protected Document loadDocument(URL source) throws IOException {
82 super.loadDocument(source); // prepares super.getSourceFile() and
83 // super.getInput()
84
85 InputStream in = super.getInput();
86 ZipInputStream zipIn = null;
87 try {
88 zipIn = new ZipInputStream(in);
076caecc
NR
89 tmpDir = Instance.getInstance().getTempFiles()
90 .createTempDir("fanfic-reader-parser");
7445f856
NR
91 File tmp = new File(tmpDir, "file.txt");
92 File tmpInfo = new File(tmpDir, "file.info");
93
94 fakeSource = tmp.toURI().toURL();
95 Image cover = null;
96
97 String url;
98 try {
99 url = getSource().toURI().toURL().toString();
100 } catch (URISyntaxException e1) {
101 url = getSource().toString();
102 }
103 String title = null;
104 String author = null;
105
076caecc
NR
106 for (ZipEntry entry = zipIn
107 .getNextEntry(); entry != null; entry = zipIn
108 .getNextEntry()) {
7445f856
NR
109 if (!entry.isDirectory()
110 && entry.getName().startsWith(getDataPrefix())) {
111 String entryLName = entry.getName().toLowerCase();
15da4d0a 112 entryLName = entryLName.substring(getDataPrefix().length());
7445f856
NR
113
114 boolean imageEntry = false;
8d59ce07 115 for (String ext : bsImages.getImageExt(false)) {
7445f856
NR
116 if (entryLName.endsWith(ext)) {
117 imageEntry = true;
118 }
08fe2e33 119 }
08fe2e33 120
15da4d0a 121 if (entryLName.equals("version")) {
7445f856
NR
122 // Nothing to do for now ("first"
123 // version is 3.0)
124 } else if (entryLName.endsWith(".info")) {
125 // Info file
126 IOUtils.write(zipIn, tmpInfo);
127 } else if (imageEntry) {
128 // Cover
15da4d0a 129 if (getCover() && cover == null) {
7445f856
NR
130 try {
131 cover = new Image(zipIn);
132 } catch (Exception e) {
076caecc
NR
133 Instance.getInstance().getTraceHandler()
134 .error(e);
7445f856 135 }
08fe2e33 136 }
15da4d0a 137 } else if (entryLName.equals("url")) {
7445f856
NR
138 String[] descArray = StringUtils
139 .unhtml(IOUtils.readSmallStream(zipIn)).trim()
140 .split("\n");
141 if (descArray.length > 0) {
142 url = descArray[0].trim();
143 }
15da4d0a 144 } else if (entryLName.endsWith(".desc")) {
076caecc
NR
145 // // For old files
146 // if (this.desc != null) {
147 // this.desc = IOUtils.readSmallStream(zipIn).trim();
148 // }
15da4d0a 149 } else if (entryLName.equals("summary")) {
7445f856
NR
150 String[] descArray = StringUtils
151 .unhtml(IOUtils.readSmallStream(zipIn)).trim()
152 .split("\n");
153 int skip = 0;
154 if (descArray.length > 1) {
155 title = descArray[0].trim();
156 skip = 1;
157 if (descArray.length > 2
158 && descArray[1].startsWith("©")) {
159 author = descArray[1].substring(1).trim();
160 skip = 2;
161 }
162 }
076caecc
NR
163 // this.desc = "";
164 // for (int i = skip; i < descArray.length; i++) {
165 // this.desc += descArray[i].trim() + "\n";
166 // }
167 //
168 // this.desc = this.desc.trim();
7445f856
NR
169 } else {
170 // Hopefully the data file
171 IOUtils.write(zipIn, tmp);
172 }
08fe2e33
NR
173 }
174 }
08fe2e33 175
15da4d0a 176 if (requireInfo() && !tmp.exists()) {
7445f856
NR
177 throw new IOException(
178 "file not supported (maybe not created with this program or corrupt)");
179 }
68686a37 180
7445f856 181 if (tmp.exists()) {
67837328 182 this.fakeIn = new MarkableFileInputStream(tmp);
68686a37 183 }
7445f856
NR
184
185 if (tmpInfo.exists()) {
186 meta = InfoReader.readMeta(tmpInfo, true);
7445f856
NR
187 tmpInfo.delete();
188 } else {
189 if (title == null || title.isEmpty()) {
190 title = getSourceFileOriginal().getName();
15da4d0a
NR
191 String exts[] = new String[] {".epub", ".cbz"};
192 for (String ext : exts) {
193 if (title.toLowerCase().endsWith(ext)) {
194 title = title.substring(0,
195 title.length() - ext.length());
196 }
7445f856
NR
197 }
198 title = URLDecoder.decode(title, "UTF-8").trim();
778d8d85 199 }
778d8d85 200
7445f856
NR
201 meta = new MetaData();
202 meta.setLang("en");
15da4d0a 203 meta.setTags(Arrays.asList("[no_info]"));
727108fe 204 meta.setSource(getType().getSourceName());
7445f856
NR
205 meta.setUuid(url);
206 meta.setUrl(url);
207 meta.setTitle(title);
208 meta.setAuthor(author);
209 meta.setImageDocument(isImagesDocumentByDefault());
15da4d0a
NR
210
211 InfoReader.completeMeta(tmp, meta);
7445f856 212 }
bb7021f2
NR
213
214 if (meta.getCover() == null) {
215 if (cover != null) {
216 meta.setCover(cover);
217 } else {
076caecc
NR
218 meta.setCover(InfoReader.getCoverByName(
219 getSourceFileOriginal().toURI().toURL()));
bb7021f2
NR
220 }
221 }
7445f856
NR
222 } finally {
223 if (zipIn != null) {
224 zipIn.close();
225 }
226 if (in != null) {
227 in.close();
228 }
08fe2e33 229 }
7445f856
NR
230
231 return null;
08fe2e33
NR
232 }
233
234 @Override
0ffa4754 235 protected void close() {
2aac79c7
NR
236 if (tmpDir != null) {
237 IOUtils.deltree(tmpDir);
08fe2e33
NR
238 }
239
2aac79c7 240 tmpDir = null;
dea63313 241
68686a37 242 super.close();
08fe2e33
NR
243 }
244
245 protected String getDataPrefix() {
246 return "DATA/";
247 }
248
249 protected boolean requireInfo() {
250 return true;
251 }
252
253 protected boolean getCover() {
254 return true;
255 }
e4fa48a0
NR
256
257 protected boolean isImagesDocumentByDefault() {
258 return false;
259 }
08fe2e33 260}