8f6e9c29a24bfd128c64e5fe69e90c98a53ac4e3
[fanfix.git] / src / be / nikiroo / fanfix / library / CacheLibrary.java
1 package be.nikiroo.fanfix.library;
2
3 import java.io.File;
4 import java.io.IOException;
5 import java.net.URL;
6 import java.util.List;
7
8 import be.nikiroo.fanfix.Instance;
9 import be.nikiroo.fanfix.bundles.UiConfig;
10 import be.nikiroo.fanfix.data.MetaData;
11 import be.nikiroo.fanfix.data.Story;
12 import be.nikiroo.utils.Image;
13 import be.nikiroo.utils.Progress;
14
15 /**
16 * This library will cache another pre-existing {@link BasicLibrary}.
17 *
18 * @author niki
19 */
20 public class CacheLibrary extends BasicLibrary {
21 private List<MetaData> metas;
22 private BasicLibrary lib;
23 private LocalLibrary cacheLib;
24
25 /**
26 * Create a cache library around the given one.
27 * <p>
28 * It will return the same result, but those will be saved to disk at the
29 * same time to be fetched quicker the next time.
30 *
31 * @param cacheDir
32 * the cache directory where to save the files to disk
33 * @param lib
34 * the original library to wrap
35 */
36 public CacheLibrary(File cacheDir, BasicLibrary lib) {
37 this.cacheLib = new LocalLibrary(cacheDir, Instance.getUiConfig()
38 .getString(UiConfig.GUI_NON_IMAGES_DOCUMENT_TYPE), Instance
39 .getUiConfig().getString(UiConfig.GUI_IMAGES_DOCUMENT_TYPE),
40 true);
41 this.lib = lib;
42 }
43
44 @Override
45 public String getLibraryName() {
46 return lib.getLibraryName();
47 }
48
49 @Override
50 public Status getStatus() {
51 return lib.getStatus();
52 }
53
54 @Override
55 protected List<MetaData> getMetas(Progress pg) {
56 if (pg == null) {
57 pg = new Progress();
58 }
59
60 if (metas == null) {
61 metas = lib.getMetas(pg);
62 }
63
64 pg.done();
65 return metas;
66 }
67
68 @Override
69 public synchronized MetaData getInfo(String luid) {
70 MetaData info = cacheLib.getInfo(luid);
71 if (info == null) {
72 info = lib.getInfo(luid);
73 }
74
75 return info;
76 }
77
78 @Override
79 public synchronized Story getStory(String luid, MetaData meta, Progress pg) {
80 if (pg == null) {
81 pg = new Progress();
82 }
83
84 Progress pgImport = new Progress();
85 Progress pgGet = new Progress();
86
87 pg.setMinMax(0, 4);
88 pg.addProgress(pgImport, 3);
89 pg.addProgress(pgGet, 1);
90
91 if (!isCached(luid)) {
92 try {
93 cacheLib.imprt(lib, luid, pgImport);
94 updateInfo(cacheLib.getInfo(luid));
95 pgImport.done();
96 } catch (IOException e) {
97 Instance.getTraceHandler().error(e);
98 }
99
100 pgImport.done();
101 pgGet.done();
102 }
103
104 String type = cacheLib.getOutputType(meta.isImageDocument());
105 MetaData cachedMeta = meta.clone();
106 cachedMeta.setType(type);
107
108 return cacheLib.getStory(luid, cachedMeta, pg);
109 }
110
111 @Override
112 public synchronized File getFile(final String luid, Progress pg) {
113 if (pg == null) {
114 pg = new Progress();
115 }
116
117 Progress pgGet = new Progress();
118 Progress pgRecall = new Progress();
119
120 pg.setMinMax(0, 5);
121 pg.addProgress(pgGet, 4);
122 pg.addProgress(pgRecall, 1);
123
124 if (!isCached(luid)) {
125 getStory(luid, pgGet);
126 pgGet.done();
127 }
128
129 File file = cacheLib.getFile(luid, pgRecall);
130 pgRecall.done();
131
132 pg.done();
133 return file;
134 }
135
136 @Override
137 public Image getCover(final String luid) {
138 if (isCached(luid)) {
139 return cacheLib.getCover(luid);
140 }
141
142 // We could update the cache here, but it's not easy
143 return lib.getCover(luid);
144 }
145
146 @Override
147 public Image getSourceCover(String source) {
148 Image custom = getCustomSourceCover(source);
149 if (custom != null) {
150 return custom;
151 }
152
153 Image cached = cacheLib.getSourceCover(source);
154 if (cached != null) {
155 return cached;
156 }
157
158 return lib.getSourceCover(source);
159 }
160
161 @Override
162 public Image getAuthorCover(String author) {
163 Image custom = getCustomAuthorCover(author);
164 if (custom != null) {
165 return custom;
166 }
167
168 Image cached = cacheLib.getAuthorCover(author);
169 if (cached != null) {
170 return cached;
171 }
172
173 return lib.getAuthorCover(author);
174 }
175
176 @Override
177 public Image getCustomSourceCover(String source) {
178 Image custom = cacheLib.getCustomSourceCover(source);
179 if (custom == null) {
180 custom = lib.getCustomSourceCover(source);
181 if (custom != null) {
182 cacheLib.setSourceCover(source, custom);
183 }
184 }
185
186 return custom;
187 }
188
189 @Override
190 public Image getCustomAuthorCover(String author) {
191 Image custom = cacheLib.getCustomAuthorCover(author);
192 if (custom == null) {
193 custom = lib.getCustomAuthorCover(author);
194 if (custom != null) {
195 cacheLib.setAuthorCover(author, custom);
196 }
197 }
198
199 return custom;
200 }
201
202 @Override
203 public void setSourceCover(String source, String luid) {
204 lib.setSourceCover(source, luid);
205 cacheLib.setSourceCover(source, getCover(luid));
206 }
207
208 @Override
209 public void setAuthorCover(String author, String luid) {
210 lib.setAuthorCover(author, luid);
211 cacheLib.setAuthorCover(author, getCover(luid));
212 }
213
214 @Override
215 protected void updateInfo(MetaData meta) {
216 if (meta != null && metas != null) {
217 for (int i = 0; i < metas.size(); i++) {
218 if (metas.get(i).getLuid().equals(meta.getLuid())) {
219 metas.set(i, meta);
220 }
221 }
222 }
223
224 cacheLib.updateInfo(meta);
225 lib.updateInfo(meta);
226 }
227
228 @Override
229 protected void invalidateInfo(String luid) {
230 if (luid == null) {
231 metas = null;
232 } else if (metas != null) {
233 for (int i = 0; i < metas.size(); i++) {
234 if (metas.get(i).getLuid().equals(luid)) {
235 metas.remove(i--);
236 }
237 }
238 }
239
240 cacheLib.invalidateInfo(luid);
241 lib.invalidateInfo(luid);
242 }
243
244 @Override
245 public synchronized Story save(Story story, String luid, Progress pg)
246 throws IOException {
247 Progress pgLib = new Progress();
248 Progress pgCacheLib = new Progress();
249
250 if (pg == null) {
251 pg = new Progress();
252 }
253
254 pg.setMinMax(0, 2);
255 pg.addProgress(pgLib, 1);
256 pg.addProgress(pgCacheLib, 1);
257
258 story = lib.save(story, luid, pgLib);
259 story = cacheLib.save(story, story.getMeta().getLuid(), pgCacheLib);
260
261 updateInfo(story.getMeta());
262
263 return story;
264 }
265
266 @Override
267 public synchronized void delete(String luid) throws IOException {
268 if (isCached(luid)) {
269 cacheLib.delete(luid);
270 }
271 lib.delete(luid);
272
273 invalidateInfo(luid);
274 }
275
276 @Override
277 protected synchronized void changeSTA(String luid, String newSource,
278 String newTitle, String newAuthor, Progress pg) throws IOException {
279 if (pg == null) {
280 pg = new Progress();
281 }
282
283 Progress pgCache = new Progress();
284 Progress pgOrig = new Progress();
285 pg.setMinMax(0, 2);
286 pg.addProgress(pgCache, 1);
287 pg.addProgress(pgOrig, 1);
288
289 MetaData meta = getInfo(luid);
290 if (meta == null) {
291 throw new IOException("Story not found: " + luid);
292 }
293
294 if (isCached(luid)) {
295 cacheLib.changeSTA(luid, newSource, newTitle, newAuthor, pgCache);
296 }
297 pgCache.done();
298
299 lib.changeSTA(luid, newSource, newTitle, newAuthor, pgOrig);
300 pgOrig.done();
301
302 meta.setSource(newSource);
303 meta.setTitle(newTitle);
304 meta.setAuthor(newAuthor);
305 pg.done();
306
307 invalidateInfo(luid);
308 }
309
310 /**
311 * Check if the {@link Story} denoted by this Library UID is present in the
312 * cache.
313 *
314 * @param luid
315 * the Library UID
316 *
317 * @return TRUE if it is
318 */
319 public boolean isCached(String luid) {
320 return cacheLib.getInfo(luid) != null;
321 }
322
323 /**
324 * Clear the {@link Story} from the cache.
325 * <p>
326 * The next time we try to retrieve the {@link Story}, it may be required to
327 * cache it again.
328 *
329 * @param luid
330 * the story to clear
331 *
332 * @throws IOException
333 * in case of I/O error
334 */
335 public void clearFromCache(String luid) throws IOException {
336 if (isCached(luid)) {
337 cacheLib.delete(luid);
338 }
339 }
340
341 @Override
342 public Story imprt(URL url, Progress pg) throws IOException {
343 if (pg == null) {
344 pg = new Progress();
345 }
346
347 Progress pgImprt = new Progress();
348 Progress pgCache = new Progress();
349 pg.setMinMax(0, 10);
350 pg.addProgress(pgImprt, 7);
351 pg.addProgress(pgCache, 3);
352
353 Story story = lib.imprt(url, pgImprt);
354 cacheLib.save(story, story.getMeta().getLuid(), pgCache);
355
356 updateInfo(story.getMeta());
357
358 pg.done();
359 return story;
360 }
361
362 // All the following methods are only used by Save and Delete in
363 // BasicLibrary:
364
365 @Override
366 protected int getNextId() {
367 throw new java.lang.InternalError("Should not have been called");
368 }
369
370 @Override
371 protected void doDelete(String luid) throws IOException {
372 throw new java.lang.InternalError("Should not have been called");
373 }
374
375 @Override
376 protected Story doSave(Story story, Progress pg) throws IOException {
377 throw new java.lang.InternalError("Should not have been called");
378 }
379 }