DownloadManager.java 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380
  1. package eu.kanade.mangafeed.data.download;
  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.net.MalformedURLException;
  13. import java.net.URL;
  14. import java.util.List;
  15. import eu.kanade.mangafeed.data.database.models.Chapter;
  16. import eu.kanade.mangafeed.data.database.models.Manga;
  17. import eu.kanade.mangafeed.data.download.model.Download;
  18. import eu.kanade.mangafeed.data.download.model.DownloadQueue;
  19. import eu.kanade.mangafeed.data.source.model.Page;
  20. import eu.kanade.mangafeed.data.preference.PreferencesHelper;
  21. import eu.kanade.mangafeed.data.source.SourceManager;
  22. import eu.kanade.mangafeed.event.DownloadChaptersEvent;
  23. import eu.kanade.mangafeed.data.source.base.Source;
  24. import eu.kanade.mangafeed.util.DiskUtils;
  25. import eu.kanade.mangafeed.util.DynamicConcurrentMergeOperator;
  26. import rx.Observable;
  27. import rx.Subscription;
  28. import rx.android.schedulers.AndroidSchedulers;
  29. import rx.schedulers.Schedulers;
  30. import rx.subjects.BehaviorSubject;
  31. import rx.subjects.PublishSubject;
  32. import timber.log.Timber;
  33. public class DownloadManager {
  34. private Context context;
  35. private SourceManager sourceManager;
  36. private PreferencesHelper preferences;
  37. private Gson gson;
  38. private PublishSubject<Download> downloadsQueueSubject;
  39. private BehaviorSubject<Integer> threadsNumber;
  40. private Subscription downloadsSubscription;
  41. private Subscription threadsNumberSubscription;
  42. private DownloadQueue queue;
  43. private volatile boolean isQueuePaused;
  44. private volatile boolean isRunning;
  45. public static final String PAGE_LIST_FILE = "index.json";
  46. public DownloadManager(Context context, SourceManager sourceManager, PreferencesHelper preferences) {
  47. this.context = context;
  48. this.sourceManager = sourceManager;
  49. this.preferences = preferences;
  50. gson = new Gson();
  51. queue = new DownloadQueue();
  52. downloadsQueueSubject = PublishSubject.create();
  53. threadsNumber = BehaviorSubject.create();
  54. }
  55. public void initializeSubscriptions() {
  56. if (downloadsSubscription != null && !downloadsSubscription.isUnsubscribed())
  57. downloadsSubscription.unsubscribe();
  58. if (threadsNumberSubscription != null && !threadsNumberSubscription.isUnsubscribed())
  59. threadsNumberSubscription.unsubscribe();
  60. threadsNumberSubscription = preferences.getDownloadTheadsObservable()
  61. .filter(n -> !isQueuePaused)
  62. .doOnNext(n -> isQueuePaused = (n == 0))
  63. .subscribe(threadsNumber::onNext);
  64. downloadsSubscription = downloadsQueueSubject
  65. .lift(new DynamicConcurrentMergeOperator<>(this::downloadChapter, threadsNumber))
  66. .onBackpressureBuffer()
  67. .observeOn(AndroidSchedulers.mainThread())
  68. .subscribe(finished -> {
  69. if (finished) {
  70. DownloadService.stop(context);
  71. }
  72. }, e -> Timber.e(e.fillInStackTrace(), e.getMessage()));
  73. isRunning = true;
  74. }
  75. public void destroySubscriptions() {
  76. isRunning = false;
  77. if (downloadsSubscription != null && !downloadsSubscription.isUnsubscribed()) {
  78. downloadsSubscription.unsubscribe();
  79. downloadsSubscription = null;
  80. }
  81. if (threadsNumberSubscription != null && !threadsNumberSubscription.isUnsubscribed()) {
  82. threadsNumberSubscription.unsubscribe();
  83. threadsNumberSubscription = null;
  84. }
  85. }
  86. // Create a download object for every chapter in the event and add them to the downloads queue
  87. public void onDownloadChaptersEvent(DownloadChaptersEvent event) {
  88. final Manga manga = event.getManga();
  89. final Source source = sourceManager.get(manga.source);
  90. for (Chapter chapter : event.getChapters()) {
  91. Download download = new Download(source, manga, chapter);
  92. if (!isChapterDownloaded(download)) {
  93. queue.add(download);
  94. if (isRunning) downloadsQueueSubject.onNext(download);
  95. }
  96. }
  97. }
  98. // Check if a chapter is already downloaded
  99. private boolean isChapterDownloaded(Download download) {
  100. // If the chapter is already queued, don't add it again
  101. for (Download queuedDownload : queue.get()) {
  102. if (download.chapter.id.equals(queuedDownload.chapter.id))
  103. return true;
  104. }
  105. // Add the directory to the download object for future access
  106. download.directory = getAbsoluteChapterDirectory(download);
  107. // If the directory doesn't exist, the chapter isn't downloaded.
  108. if (!download.directory.exists()) {
  109. return false;
  110. }
  111. // If the page list doesn't exist, the chapter isn't download (or maybe it's,
  112. // but we consider it's not)
  113. List<Page> savedPages = getSavedPageList(download);
  114. if (savedPages == null)
  115. return false;
  116. // Add the page list to the download object for future access
  117. download.pages = savedPages;
  118. // If the number of files matches the number of pages, the chapter is downloaded.
  119. // We have the index file, so we check one file more
  120. return savedPages.size() + 1 == download.directory.listFiles().length;
  121. }
  122. // Download the entire chapter
  123. private Observable<Boolean> downloadChapter(Download download) {
  124. try {
  125. DiskUtils.createDirectory(download.directory);
  126. } catch (IOException e) {
  127. Timber.e(e.getMessage());
  128. }
  129. Observable<List<Page>> pageListObservable = download.pages == null ?
  130. // Pull page list from network and add them to download object
  131. download.source
  132. .pullPageListFromNetwork(download.chapter.url)
  133. .doOnNext(pages -> download.pages = pages)
  134. .doOnNext(pages -> savePageList(download)) :
  135. // Or if the page list already exists, start from the file
  136. Observable.just(download.pages);
  137. return pageListObservable
  138. .subscribeOn(Schedulers.io())
  139. .doOnNext(pages -> download.downloadedImages = 0)
  140. .doOnNext(pages -> download.setStatus(Download.DOWNLOADING))
  141. // Get all the URLs to the source images, fetch pages if necessary
  142. .flatMap(pageList -> Observable.merge(
  143. Observable.from(pageList).filter(page -> page.getImageUrl() != null),
  144. download.source.getRemainingImageUrlsFromPageList(pageList)))
  145. // Start downloading images, consider we can have downloaded images already
  146. .concatMap(page -> getDownloadedImage(page, download.source, download.directory))
  147. .doOnNext(p -> download.downloadedImages++)
  148. // Do after download completes
  149. .doOnCompleted(() -> onDownloadCompleted(download))
  150. .toList()
  151. .flatMap(pages -> Observable.just(areAllDownloadsFinished()));
  152. }
  153. // Get downloaded image if exists, otherwise download it with the method below
  154. public Observable<Page> getDownloadedImage(final Page page, Source source, File chapterDir) {
  155. Observable<Page> pageObservable = Observable.just(page);
  156. if (page.getImageUrl() == null)
  157. return pageObservable;
  158. String imageFilename = getImageFilename(page);
  159. File imagePath = new File(chapterDir, imageFilename);
  160. if (!isImageDownloaded(imagePath)) {
  161. page.setStatus(Page.DOWNLOAD_IMAGE);
  162. pageObservable = downloadImage(page, source, chapterDir, imageFilename);
  163. }
  164. return pageObservable
  165. // When the image is ready, set image path, progress (just in case) and status
  166. .doOnNext(p -> {
  167. p.setImagePath(imagePath.getAbsolutePath());
  168. p.setProgress(100);
  169. p.setStatus(Page.READY);
  170. })
  171. // If the download fails, mark this page as error
  172. .doOnError(e -> page.setStatus(Page.ERROR))
  173. // Allow to download the remaining images
  174. .onErrorResumeNext(e -> Observable.just(page));
  175. }
  176. // Download the image and save it to the filesystem
  177. private Observable<Page> downloadImage(final Page page, Source source, File chapterDir, String imageFilename) {
  178. return source.getImageProgressResponse(page)
  179. .flatMap(resp -> {
  180. try {
  181. DiskUtils.saveBufferedSourceToDirectory(resp.body().source(), chapterDir, imageFilename);
  182. } catch (IOException e) {
  183. Timber.e(e.fillInStackTrace(), e.getMessage());
  184. throw new IllegalStateException("Unable to save image");
  185. }
  186. return Observable.just(page);
  187. });
  188. }
  189. // Get the filename for an image given the page
  190. private String getImageFilename(Page page) {
  191. String url;
  192. try {
  193. url = new URL(page.getImageUrl()).getPath();
  194. } catch (MalformedURLException e) {
  195. url = page.getImageUrl();
  196. }
  197. return url.substring(
  198. url.lastIndexOf("/") + 1,
  199. url.length());
  200. }
  201. private boolean isImageDownloaded(File imagePath) {
  202. return imagePath.exists();
  203. }
  204. // Called when a download finishes. This doesn't mean the download was successful, so we check it
  205. private void onDownloadCompleted(final Download download) {
  206. checkDownloadIsSuccessful(download);
  207. savePageList(download);
  208. }
  209. private void checkDownloadIsSuccessful(final Download download) {
  210. int actualProgress = 0;
  211. int status = Download.DOWNLOADED;
  212. // If any page has an error, the download result will be error
  213. for (Page page : download.pages) {
  214. actualProgress += page.getProgress();
  215. if (page.getStatus() == Page.ERROR) status = Download.ERROR;
  216. }
  217. download.totalProgress = actualProgress;
  218. download.setStatus(status);
  219. }
  220. // Return the page list from the chapter's directory if it exists, null otherwise
  221. public List<Page> getSavedPageList(Source source, Manga manga, Chapter chapter) {
  222. List<Page> pages = null;
  223. File chapterDir = getAbsoluteChapterDirectory(source, manga, chapter);
  224. File pagesFile = new File(chapterDir, PAGE_LIST_FILE);
  225. JsonReader reader = null;
  226. try {
  227. if (pagesFile.exists()) {
  228. reader = new JsonReader(new FileReader(pagesFile.getAbsolutePath()));
  229. Type collectionType = new TypeToken<List<Page>>() {}.getType();
  230. pages = gson.fromJson(reader, collectionType);
  231. }
  232. } catch (FileNotFoundException e) {
  233. Timber.e(e.fillInStackTrace(), e.getMessage());
  234. } finally {
  235. if (reader != null) try { reader.close(); } catch (IOException e) { /* Do nothing */ }
  236. }
  237. return pages;
  238. }
  239. // Shortcut for the method above
  240. private List<Page> getSavedPageList(Download download) {
  241. return getSavedPageList(download.source, download.manga, download.chapter);
  242. }
  243. // Save the page list to the chapter's directory
  244. public void savePageList(Source source, Manga manga, Chapter chapter, List<Page> pages) {
  245. File chapterDir = getAbsoluteChapterDirectory(source, manga, chapter);
  246. File pagesFile = new File(chapterDir, PAGE_LIST_FILE);
  247. FileOutputStream out = null;
  248. try {
  249. out = new FileOutputStream(pagesFile);
  250. out.write(gson.toJson(pages).getBytes());
  251. out.flush();
  252. } catch (IOException e) {
  253. Timber.e(e.fillInStackTrace(), e.getMessage());
  254. } finally {
  255. if (out != null) try { out.close(); } catch (IOException e) { /* Do nothing */ }
  256. }
  257. }
  258. // Shortcut for the method above
  259. private void savePageList(Download download) {
  260. savePageList(download.source, download.manga, download.chapter, download.pages);
  261. }
  262. // Get the absolute path to the chapter directory
  263. public File getAbsoluteChapterDirectory(Source source, Manga manga, Chapter chapter) {
  264. String chapterRelativePath = source.getName() +
  265. File.separator +
  266. manga.title.replaceAll("[^a-zA-Z0-9.-]", "_") +
  267. File.separator +
  268. chapter.name.replaceAll("[^a-zA-Z0-9.-]", "_");
  269. return new File(preferences.getDownloadsDirectory(), chapterRelativePath);
  270. }
  271. // Shortcut for the method above
  272. private File getAbsoluteChapterDirectory(Download download) {
  273. return getAbsoluteChapterDirectory(download.source, download.manga, download.chapter);
  274. }
  275. public void deleteChapter(Source source, Manga manga, Chapter chapter) {
  276. File path = getAbsoluteChapterDirectory(source, manga, chapter);
  277. DiskUtils.deleteFiles(path);
  278. }
  279. public DownloadQueue getQueue() {
  280. return queue;
  281. }
  282. public boolean areAllDownloadsFinished() {
  283. for (Download download : queue.get()) {
  284. if (download.getStatus() <= Download.DOWNLOADING)
  285. return false;
  286. }
  287. return true;
  288. }
  289. public void resumeDownloads() {
  290. isQueuePaused = false;
  291. threadsNumber.onNext(preferences.getDownloadThreads());
  292. }
  293. public void pauseDownloads() {
  294. threadsNumber.onNext(0);
  295. }
  296. public boolean startDownloads() {
  297. boolean hasPendingDownloads = false;
  298. if (downloadsSubscription == null || threadsNumberSubscription == null)
  299. initializeSubscriptions();
  300. for (Download download : queue.get()) {
  301. if (download.getStatus() != Download.DOWNLOADED) {
  302. download.setStatus(Download.QUEUE);
  303. if (!hasPendingDownloads) hasPendingDownloads = true;
  304. downloadsQueueSubject.onNext(download);
  305. }
  306. }
  307. return hasPendingDownloads;
  308. }
  309. public void stopDownloads() {
  310. destroySubscriptions();
  311. for (Download download : queue.get()) {
  312. if (download.getStatus() == Download.DOWNLOADING) {
  313. download.setStatus(Download.ERROR);
  314. }
  315. }
  316. }
  317. public boolean isRunning() {
  318. return isRunning;
  319. }
  320. }