DownloadManager.java 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290
  1. package eu.kanade.mangafeed.data.helpers;
  2. import android.content.Context;
  3. import com.google.gson.Gson;
  4. import com.google.gson.reflect.TypeToken;
  5. import com.google.gson.stream.JsonReader;
  6. import java.io.File;
  7. import java.io.FileNotFoundException;
  8. import java.io.FileOutputStream;
  9. import java.io.FileReader;
  10. import java.io.IOException;
  11. import java.lang.reflect.Type;
  12. import java.util.ArrayList;
  13. import java.util.List;
  14. import eu.kanade.mangafeed.data.models.Chapter;
  15. import eu.kanade.mangafeed.data.models.Download;
  16. import eu.kanade.mangafeed.data.models.DownloadQueue;
  17. import eu.kanade.mangafeed.data.models.Manga;
  18. import eu.kanade.mangafeed.data.models.Page;
  19. import eu.kanade.mangafeed.events.DownloadChaptersEvent;
  20. import eu.kanade.mangafeed.sources.base.Source;
  21. import eu.kanade.mangafeed.util.DiskUtils;
  22. import eu.kanade.mangafeed.util.DynamicConcurrentMergeOperator;
  23. import rx.Observable;
  24. import rx.Subscription;
  25. import rx.schedulers.Schedulers;
  26. import rx.subjects.BehaviorSubject;
  27. import rx.subjects.PublishSubject;
  28. import timber.log.Timber;
  29. public class DownloadManager {
  30. private PublishSubject<DownloadChaptersEvent> downloadsSubject;
  31. private Subscription downloadSubscription;
  32. private Subscription threadNumberSubscription;
  33. private Context context;
  34. private SourceManager sourceManager;
  35. private PreferencesHelper preferences;
  36. private Gson gson;
  37. private DownloadQueue queue;
  38. public static final String PAGE_LIST_FILE = "index.json";
  39. public DownloadManager(Context context, SourceManager sourceManager, PreferencesHelper preferences) {
  40. this.context = context;
  41. this.sourceManager = sourceManager;
  42. this.preferences = preferences;
  43. this.gson = new Gson();
  44. queue = new DownloadQueue();
  45. initializeDownloadSubscription();
  46. }
  47. public PublishSubject<DownloadChaptersEvent> getDownloadsSubject() {
  48. return downloadsSubject;
  49. }
  50. private void initializeDownloadSubscription() {
  51. if (downloadSubscription != null && !downloadSubscription.isUnsubscribed()) {
  52. downloadSubscription.unsubscribe();
  53. }
  54. if (threadNumberSubscription != null && !threadNumberSubscription.isUnsubscribed())
  55. threadNumberSubscription.unsubscribe();
  56. downloadsSubject = PublishSubject.create();
  57. BehaviorSubject<Integer> threads = BehaviorSubject.create();
  58. threadNumberSubscription = preferences.getDownloadTheadsObs()
  59. .subscribe(threads::onNext);
  60. // Listen for download events, add them to queue and download
  61. downloadSubscription = downloadsSubject
  62. .subscribeOn(Schedulers.io())
  63. .flatMap(this::prepareDownloads)
  64. .lift(new DynamicConcurrentMergeOperator<>(this::downloadChapter, threads))
  65. .onBackpressureBuffer()
  66. .subscribe(page -> {},
  67. e -> Timber.e(e.fillInStackTrace(), e.getMessage()));
  68. }
  69. // Create a download object for every chapter and add it to the downloads queue
  70. private Observable<Download> prepareDownloads(DownloadChaptersEvent event) {
  71. final Manga manga = event.getManga();
  72. final Source source = sourceManager.get(manga.source);
  73. List<Download> downloads = new ArrayList<>();
  74. for (Chapter chapter : event.getChapters()) {
  75. Download download = new Download(source, manga, chapter);
  76. if (!isChapterDownloaded(download)) {
  77. queue.add(download);
  78. downloads.add(download);
  79. }
  80. }
  81. return Observable.from(downloads);
  82. }
  83. // Check if a chapter is already downloaded
  84. private boolean isChapterDownloaded(Download download) {
  85. // If the chapter is already queued, don't add it again
  86. for (Download queuedDownload : queue.get()) {
  87. if (download.chapter.id == queuedDownload.chapter.id)
  88. return true;
  89. }
  90. // Add the directory to the download object for future access
  91. download.directory = getAbsoluteChapterDirectory(download);
  92. // If the directory doesn't exist, the chapter isn't downloaded. Create it in this case
  93. if (!download.directory.exists()) {
  94. // FIXME Sometimes it's failing to create the directory... My fault?
  95. try {
  96. DiskUtils.createDirectory(download.directory);
  97. } catch (IOException e) {
  98. Timber.e("Unable to create directory for chapter");
  99. }
  100. return false;
  101. }
  102. // If the page list doesn't exist, the chapter isn't download (or maybe it's,
  103. // but we consider it's not)
  104. List<Page> savedPages = getSavedPageList(download);
  105. if (savedPages == null)
  106. return false;
  107. // Add the page list to the download object for future access
  108. download.pages = savedPages;
  109. // If the number of files matches the number of pages, the chapter is downloaded.
  110. // We have the index file, so we check one file more
  111. return savedPages.size() + 1 == download.directory.listFiles().length;
  112. }
  113. // Download the entire chapter
  114. private Observable<Page> downloadChapter(Download download) {
  115. Observable<List<Page>> pageListObservable = download.pages == null ?
  116. // Pull page list from network and add them to download object
  117. download.source
  118. .pullPageListFromNetwork(download.chapter.url)
  119. .doOnNext(pages -> download.pages = pages)
  120. .doOnNext(pages -> savePageList(download)) :
  121. // Or if the file exists, start from here
  122. Observable.just(download.pages);
  123. return pageListObservable
  124. .subscribeOn(Schedulers.io())
  125. .doOnNext(pages -> download.setStatus(Download.DOWNLOADING))
  126. // Get all the URLs to the source images, fetch pages if necessary
  127. .flatMap(pageList -> Observable.merge(
  128. Observable.from(pageList).filter(page -> page.getImageUrl() != null),
  129. download.source.getRemainingImageUrlsFromPageList(pageList)))
  130. // Start downloading images, consider we can have downloaded images already
  131. .concatMap(page -> getDownloadedImage(page, download.source, download.directory))
  132. // Do after download completes
  133. .doOnCompleted(() -> onChapterDownloaded(download));
  134. }
  135. // Get downloaded image if exists, otherwise download it with the method below
  136. public Observable<Page> getDownloadedImage(final Page page, Source source, File chapterDir) {
  137. Observable<Page> obs = Observable.just(page);
  138. if (page.getImageUrl() == null)
  139. return obs;
  140. String imageFilename = getImageFilename(page);
  141. File imagePath = new File(chapterDir, imageFilename);
  142. if (!isImageDownloaded(imagePath)) {
  143. page.setStatus(Page.DOWNLOAD_IMAGE);
  144. obs = downloadImage(page, source, chapterDir, imageFilename);
  145. }
  146. return obs.flatMap(p -> {
  147. page.setImagePath(imagePath.getAbsolutePath());
  148. page.setStatus(Page.READY);
  149. return Observable.just(page);
  150. }).onErrorResumeNext(e -> {
  151. page.setStatus(Page.ERROR);
  152. return Observable.just(page);
  153. });
  154. }
  155. // Download the image
  156. private Observable<Page> downloadImage(final Page page, Source source, File chapterDir, String imageFilename) {
  157. return source.getImageProgressResponse(page)
  158. .flatMap(resp -> {
  159. try {
  160. DiskUtils.saveBufferedSourceToDirectory(resp.body().source(), chapterDir, imageFilename);
  161. } catch (IOException e) {
  162. Timber.e(e.fillInStackTrace(), e.getMessage());
  163. throw new IllegalStateException("Unable to save image");
  164. }
  165. return Observable.just(page);
  166. });
  167. }
  168. // Get the filename for an image given the page
  169. private String getImageFilename(Page page) {
  170. return page.getImageUrl().substring(
  171. page.getImageUrl().lastIndexOf("/") + 1,
  172. page.getImageUrl().length());
  173. }
  174. private boolean isImageDownloaded(File imagePath) {
  175. return imagePath.exists() && !imagePath.isDirectory();
  176. }
  177. private void onChapterDownloaded(final Download download) {
  178. download.setStatus(Download.DOWNLOADED);
  179. download.totalProgress = download.pages.size() * 100;
  180. savePageList(download.source, download.manga, download.chapter, download.pages);
  181. }
  182. // Return the page list from the chapter's directory if it exists, null otherwise
  183. public List<Page> getSavedPageList(Source source, Manga manga, Chapter chapter) {
  184. File chapterDir = getAbsoluteChapterDirectory(source, manga, chapter);
  185. File pagesFile = new File(chapterDir, PAGE_LIST_FILE);
  186. try {
  187. if (pagesFile.exists()) {
  188. JsonReader reader = new JsonReader(new FileReader(pagesFile.getAbsolutePath()));
  189. Type collectionType = new TypeToken<List<Page>>() {}.getType();
  190. return gson.fromJson(reader, collectionType);
  191. }
  192. } catch (FileNotFoundException e) {
  193. Timber.e(e.fillInStackTrace(), e.getMessage());
  194. }
  195. return null;
  196. }
  197. // Shortcut for the method above
  198. private List<Page> getSavedPageList(Download download) {
  199. return getSavedPageList(download.source, download.manga, download.chapter);
  200. }
  201. // Save the page list to the chapter's directory
  202. public void savePageList(Source source, Manga manga, Chapter chapter, List<Page> pages) {
  203. File chapterDir = getAbsoluteChapterDirectory(source, manga, chapter);
  204. File pagesFile = new File(chapterDir, PAGE_LIST_FILE);
  205. FileOutputStream out;
  206. try {
  207. out = new FileOutputStream(pagesFile);
  208. out.write(gson.toJson(pages).getBytes());
  209. out.flush();
  210. out.close();
  211. } catch (Exception e) {
  212. Timber.e(e.fillInStackTrace(), e.getMessage());
  213. }
  214. }
  215. // Shortcut for the method above
  216. private void savePageList(Download download) {
  217. savePageList(download.source, download.manga, download.chapter, download.pages);
  218. }
  219. // Get the absolute path to the chapter directory
  220. public File getAbsoluteChapterDirectory(Source source, Manga manga, Chapter chapter) {
  221. String chapterRelativePath = source.getName() +
  222. File.separator +
  223. manga.title.replaceAll("[^a-zA-Z0-9.-]", "_") +
  224. File.separator +
  225. chapter.name.replaceAll("[^a-zA-Z0-9.-]", "_");
  226. return new File(preferences.getDownloadsDirectory(), chapterRelativePath);
  227. }
  228. // Shortcut for the method above
  229. private File getAbsoluteChapterDirectory(Download download) {
  230. return getAbsoluteChapterDirectory(download.source, download.manga, download.chapter);
  231. }
  232. public void deleteChapter(Source source, Manga manga, Chapter chapter) {
  233. File path = getAbsoluteChapterDirectory(source, manga, chapter);
  234. DiskUtils.deleteFiles(path);
  235. }
  236. public DownloadQueue getQueue() {
  237. return queue;
  238. }
  239. }