CacheManager.java 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248
  1. package eu.kanade.mangafeed.data.caches;
  2. import android.content.Context;
  3. import com.bumptech.glide.Glide;
  4. import com.bumptech.glide.request.FutureTarget;
  5. import com.bumptech.glide.request.target.Target;
  6. import com.google.gson.Gson;
  7. import com.google.gson.reflect.TypeToken;
  8. import com.jakewharton.disklrucache.DiskLruCache;
  9. import com.squareup.okhttp.Response;
  10. import java.io.BufferedOutputStream;
  11. import java.io.File;
  12. import java.io.IOException;
  13. import java.io.OutputStream;
  14. import java.lang.reflect.Type;
  15. import java.util.List;
  16. import java.util.concurrent.ExecutionException;
  17. import java.util.concurrent.TimeUnit;
  18. import java.util.concurrent.TimeoutException;
  19. import eu.kanade.mangafeed.data.models.Page;
  20. import eu.kanade.mangafeed.util.DiskUtils;
  21. import okio.BufferedSink;
  22. import okio.Okio;
  23. import rx.Observable;
  24. public class CacheManager {
  25. private static final String PARAMETER_CACHE_DIRECTORY = "chapter_disk_cache";
  26. private static final int PARAMETER_APP_VERSION = 1;
  27. private static final int PARAMETER_VALUE_COUNT = 1;
  28. private static final long PARAMETER_CACHE_SIZE = 10 * 1024 * 1024;
  29. private static final int READ_TIMEOUT = 60;
  30. private Context mContext;
  31. private Gson mGson;
  32. private DiskLruCache mDiskCache;
  33. public CacheManager(Context context) {
  34. mContext = context;
  35. mGson = new Gson();
  36. try {
  37. mDiskCache = DiskLruCache.open(
  38. new File(context.getCacheDir(), PARAMETER_CACHE_DIRECTORY),
  39. PARAMETER_APP_VERSION,
  40. PARAMETER_VALUE_COUNT,
  41. PARAMETER_CACHE_SIZE
  42. );
  43. } catch (IOException e) {
  44. // Do Nothing.
  45. }
  46. }
  47. public Observable<File> cacheImagesFromUrls(final List<String> imageUrls) {
  48. return Observable.create(subscriber -> {
  49. try {
  50. for (String imageUrl : imageUrls) {
  51. if (!subscriber.isUnsubscribed()) {
  52. subscriber.onNext(cacheImageFromUrl(imageUrl));
  53. }
  54. }
  55. subscriber.onCompleted();
  56. } catch (Throwable e) {
  57. subscriber.onError(e);
  58. }
  59. });
  60. }
  61. private File cacheImageFromUrl(String imageUrl) throws InterruptedException, ExecutionException, TimeoutException {
  62. FutureTarget<File> cacheFutureTarget = Glide.with(mContext)
  63. .load(imageUrl)
  64. .downloadOnly(Target.SIZE_ORIGINAL, Target.SIZE_ORIGINAL);
  65. return cacheFutureTarget.get(READ_TIMEOUT, TimeUnit.SECONDS);
  66. }
  67. public Observable<Boolean> clearImageCache() {
  68. return Observable.create(subscriber -> {
  69. try {
  70. subscriber.onNext(clearImageCacheImpl());
  71. subscriber.onCompleted();
  72. } catch (Throwable e) {
  73. subscriber.onError(e);
  74. }
  75. });
  76. }
  77. private boolean clearImageCacheImpl() {
  78. boolean isSuccessful = true;
  79. File imageCacheDirectory = Glide.getPhotoCacheDir(mContext);
  80. if (imageCacheDirectory.isDirectory()) {
  81. for (File cachedFile : imageCacheDirectory.listFiles()) {
  82. if (!cachedFile.delete()) {
  83. isSuccessful = false;
  84. }
  85. }
  86. } else {
  87. isSuccessful = false;
  88. }
  89. File urlCacheDirectory = getCacheDir();
  90. if (urlCacheDirectory.isDirectory()) {
  91. for (File cachedFile : urlCacheDirectory.listFiles()) {
  92. if (!cachedFile.delete()) {
  93. isSuccessful = false;
  94. }
  95. }
  96. } else {
  97. isSuccessful = false;
  98. }
  99. return isSuccessful;
  100. }
  101. public Observable<List<Page>> getPageUrlsFromDiskCache(final String chapterUrl) {
  102. return Observable.create(subscriber -> {
  103. try {
  104. List<Page> pages = getPageUrlsFromDiskCacheImpl(chapterUrl);
  105. subscriber.onNext(pages);
  106. subscriber.onCompleted();
  107. } catch (Throwable e) {
  108. subscriber.onError(e);
  109. }
  110. });
  111. }
  112. private List<Page> getPageUrlsFromDiskCacheImpl(String chapterUrl) throws IOException {
  113. DiskLruCache.Snapshot snapshot = null;
  114. List<Page> pages = null;
  115. try {
  116. String key = DiskUtils.hashKeyForDisk(chapterUrl);
  117. snapshot = mDiskCache.get(key);
  118. Type collectionType = new TypeToken<List<Page>>() {}.getType();
  119. pages = mGson.fromJson(snapshot.getString(0), collectionType);
  120. } catch (IOException e) {
  121. // Do Nothing.
  122. } finally {
  123. if (snapshot != null) {
  124. snapshot.close();
  125. }
  126. }
  127. return pages;
  128. }
  129. public void putPageUrlsToDiskCache(final String chapterUrl, final List<Page> pages) {
  130. String cachedValue = mGson.toJson(pages);
  131. DiskLruCache.Editor editor = null;
  132. OutputStream outputStream = null;
  133. try {
  134. String key = DiskUtils.hashKeyForDisk(chapterUrl);
  135. editor = mDiskCache.edit(key);
  136. if (editor == null) {
  137. return;
  138. }
  139. outputStream = new BufferedOutputStream(editor.newOutputStream(0));
  140. outputStream.write(cachedValue.getBytes());
  141. outputStream.flush();
  142. mDiskCache.flush();
  143. editor.commit();
  144. } catch (Exception e) {
  145. // Do Nothing.
  146. } finally {
  147. if (editor != null) {
  148. editor.abortUnlessCommitted();
  149. }
  150. if (outputStream != null) {
  151. try {
  152. outputStream.close();
  153. } catch (IOException ignore) {
  154. // Do Nothing.
  155. }
  156. }
  157. }
  158. }
  159. public File getCacheDir() {
  160. return mDiskCache.getDirectory();
  161. }
  162. public boolean isImageInCache(final String imageUrl) {
  163. try {
  164. return mDiskCache.get(DiskUtils.hashKeyForDisk(imageUrl)) != null;
  165. } catch (IOException e) {
  166. e.printStackTrace();
  167. }
  168. return false;
  169. }
  170. public String getImagePath(final String imageUrl) {
  171. try {
  172. String imageName = DiskUtils.hashKeyForDisk(imageUrl) + ".0";
  173. File file = new File(mDiskCache.getDirectory(), imageName);
  174. return file.getCanonicalPath();
  175. } catch (IOException e) {
  176. e.printStackTrace();
  177. }
  178. return null;
  179. }
  180. public boolean putImageToDiskCache(final String imageUrl, final Response response) {
  181. DiskLruCache.Editor editor = null;
  182. BufferedSink sink = null;
  183. try {
  184. String key = DiskUtils.hashKeyForDisk(imageUrl);
  185. editor = mDiskCache.edit(key);
  186. if (editor == null) {
  187. return false;
  188. }
  189. OutputStream outputStream = new BufferedOutputStream(editor.newOutputStream(0));
  190. sink = Okio.buffer(Okio.sink(outputStream));
  191. sink.writeAll(response.body().source());
  192. sink.flush();
  193. mDiskCache.flush();
  194. editor.commit();
  195. } catch (IOException e) {
  196. e.printStackTrace();
  197. return false;
  198. } finally {
  199. if (editor != null) {
  200. editor.abortUnlessCommitted();
  201. }
  202. if (sink != null) {
  203. try {
  204. sink.close();
  205. } catch (IOException e) {
  206. e.printStackTrace();
  207. }
  208. }
  209. }
  210. return true;
  211. }
  212. }