mirror of
				https://github.com/commons-app/apps-android-commons.git
				synced 2025-10-26 12:23:58 +01:00 
			
		
		
		
	Handle failures in chunk uploads (#3916)
* Handle failures in chunk uploads * Fix failures * Upload fixed * Handle multiple file upload * Increase request timeout
This commit is contained in:
		
							parent
							
								
									0d5fa048a5
								
							
						
					
					
						commit
						6c55525a43
					
				
					 8 changed files with 69 additions and 38 deletions
				
			
		|  | @ -6,6 +6,7 @@ import java.io.IOException; | ||||||
| import java.util.Arrays; | import java.util.Arrays; | ||||||
| import java.util.Collections; | import java.util.Collections; | ||||||
| import java.util.List; | import java.util.List; | ||||||
|  | import java.util.concurrent.TimeUnit; | ||||||
| import okhttp3.Cache; | import okhttp3.Cache; | ||||||
| import okhttp3.Interceptor; | import okhttp3.Interceptor; | ||||||
| import okhttp3.OkHttpClient; | import okhttp3.OkHttpClient; | ||||||
|  | @ -36,6 +37,9 @@ public final class OkHttpConnectionFactory { | ||||||
|         return new OkHttpClient.Builder() |         return new OkHttpClient.Builder() | ||||||
|                 .cookieJar(SharedPreferenceCookieManager.getInstance()) |                 .cookieJar(SharedPreferenceCookieManager.getInstance()) | ||||||
|                 .cache(NET_CACHE) |                 .cache(NET_CACHE) | ||||||
|  |                 .connectTimeout(60, TimeUnit.SECONDS) | ||||||
|  |                 .writeTimeout(60, TimeUnit.SECONDS) | ||||||
|  |                 .readTimeout(60, TimeUnit.SECONDS) | ||||||
|                 .addInterceptor(getLoggingInterceptor()) |                 .addInterceptor(getLoggingInterceptor()) | ||||||
|                 .addInterceptor(new UnsuccessfulResponseInterceptor()) |                 .addInterceptor(new UnsuccessfulResponseInterceptor()) | ||||||
|                 .addInterceptor(new CommonHeaderRequestInterceptor()) |                 .addInterceptor(new CommonHeaderRequestInterceptor()) | ||||||
|  |  | ||||||
|  | @ -6,20 +6,20 @@ import fr.free.nrw.commons.upload.UploadResult | ||||||
| 
 | 
 | ||||||
| data class ChunkInfo( | data class ChunkInfo( | ||||||
|     val uploadResult: UploadResult, |     val uploadResult: UploadResult, | ||||||
|     val lastChunkIndex: Int, |     val indexOfNextChunkToUpload: Int, | ||||||
|     var isLastChunkUploaded: Boolean |     val totalChunks: Int | ||||||
| ) : Parcelable { | ) : Parcelable { | ||||||
|     constructor(parcel: Parcel) : this( |     constructor(parcel: Parcel) : this( | ||||||
|         parcel.readParcelable(UploadResult::class.java.classLoader), |         parcel.readParcelable(UploadResult::class.java.classLoader), | ||||||
|         parcel.readInt(), |         parcel.readInt(), | ||||||
|         parcel.readByte() != 0.toByte() |         parcel.readInt() | ||||||
|     ) { |     ) { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     override fun writeToParcel(parcel: Parcel, flags: Int) { |     override fun writeToParcel(parcel: Parcel, flags: Int) { | ||||||
|         parcel.writeParcelable(uploadResult, flags) |         parcel.writeParcelable(uploadResult, flags) | ||||||
|         parcel.writeInt(lastChunkIndex) |         parcel.writeInt(indexOfNextChunkToUpload) | ||||||
|         parcel.writeByte(if (isLastChunkUploaded) 1 else 0) |         parcel.writeInt(totalChunks) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     override fun describeContents(): Int { |     override fun describeContents(): Int { | ||||||
|  |  | ||||||
|  | @ -10,7 +10,7 @@ import fr.free.nrw.commons.contributions.ContributionDao | ||||||
|  * The database for accessing the respective DAOs |  * The database for accessing the respective DAOs | ||||||
|  * |  * | ||||||
|  */ |  */ | ||||||
| @Database(entities = [Contribution::class], version = 5, exportSchema = false) | @Database(entities = [Contribution::class], version = 6, exportSchema = false) | ||||||
| @TypeConverters(Converters::class) | @TypeConverters(Converters::class) | ||||||
| abstract class AppDatabase : RoomDatabase() { | abstract class AppDatabase : RoomDatabase() { | ||||||
|   abstract fun contributionDao(): ContributionDao |   abstract fun contributionDao(): ContributionDao | ||||||
|  |  | ||||||
|  | @ -62,7 +62,8 @@ public class NetworkingModule { | ||||||
|     public OkHttpClient provideOkHttpClient(Context context, |     public OkHttpClient provideOkHttpClient(Context context, | ||||||
|                                             HttpLoggingInterceptor httpLoggingInterceptor) { |                                             HttpLoggingInterceptor httpLoggingInterceptor) { | ||||||
|         File dir = new File(context.getCacheDir(), "okHttpCache"); |         File dir = new File(context.getCacheDir(), "okHttpCache"); | ||||||
|         return new OkHttpClient.Builder().connectTimeout(60, TimeUnit.SECONDS) |         return new OkHttpClient.Builder() | ||||||
|  |             .connectTimeout(60, TimeUnit.SECONDS) | ||||||
|             .writeTimeout(60, TimeUnit.SECONDS) |             .writeTimeout(60, TimeUnit.SECONDS) | ||||||
|                 .addInterceptor(httpLoggingInterceptor) |                 .addInterceptor(httpLoggingInterceptor) | ||||||
|             .readTimeout(60, TimeUnit.SECONDS) |             .readTimeout(60, TimeUnit.SECONDS) | ||||||
|  |  | ||||||
|  | @ -45,7 +45,7 @@ public class FileUtilsWrapper { | ||||||
|   /** |   /** | ||||||
|    * Takes a file as input and returns an Observable of files with the specified chunk size |    * Takes a file as input and returns an Observable of files with the specified chunk size | ||||||
|    */ |    */ | ||||||
|   public Observable<File> getFileChunks(Context context, File file, final int chunkSize) |   public List<File> getFileChunks(Context context, File file, final int chunkSize) | ||||||
|       throws IOException { |       throws IOException { | ||||||
|     final byte[] buffer = new byte[chunkSize]; |     final byte[] buffer = new byte[chunkSize]; | ||||||
| 
 | 
 | ||||||
|  | @ -58,7 +58,7 @@ public class FileUtilsWrapper { | ||||||
|         buffers.add(writeToFile(context, Arrays.copyOf(buffer, size), file.getName(), |         buffers.add(writeToFile(context, Arrays.copyOf(buffer, size), file.getName(), | ||||||
|             getFileExt(file.getName()))); |             getFileExt(file.getName()))); | ||||||
|       } |       } | ||||||
|       return Observable.fromIterable(buffers); |       return buffers; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -15,6 +15,10 @@ import io.reactivex.disposables.CompositeDisposable; | ||||||
| import java.io.File; | import java.io.File; | ||||||
| import java.io.IOException; | import java.io.IOException; | ||||||
| import java.util.Date; | import java.util.Date; | ||||||
|  | import java.util.HashMap; | ||||||
|  | import java.util.List; | ||||||
|  | import java.util.Map; | ||||||
|  | import java.util.concurrent.atomic.AtomicBoolean; | ||||||
| import java.util.concurrent.atomic.AtomicInteger; | import java.util.concurrent.atomic.AtomicInteger; | ||||||
| import java.util.concurrent.atomic.AtomicReference; | import java.util.concurrent.atomic.AtomicReference; | ||||||
| import javax.inject.Inject; | import javax.inject.Inject; | ||||||
|  | @ -41,7 +45,8 @@ public class UploadClient { | ||||||
|   private final PageContentsCreator pageContentsCreator; |   private final PageContentsCreator pageContentsCreator; | ||||||
|   private final FileUtilsWrapper fileUtilsWrapper; |   private final FileUtilsWrapper fileUtilsWrapper; | ||||||
|   private final Gson gson; |   private final Gson gson; | ||||||
|   private boolean pauseUploads = false; | 
 | ||||||
|  |   private Map<String, Boolean> pauseUploads; | ||||||
| 
 | 
 | ||||||
|   private final CompositeDisposable compositeDisposable = new CompositeDisposable(); |   private final CompositeDisposable compositeDisposable = new CompositeDisposable(); | ||||||
| 
 | 
 | ||||||
|  | @ -55,6 +60,7 @@ public class UploadClient { | ||||||
|     this.pageContentsCreator = pageContentsCreator; |     this.pageContentsCreator = pageContentsCreator; | ||||||
|     this.fileUtilsWrapper = fileUtilsWrapper; |     this.fileUtilsWrapper = fileUtilsWrapper; | ||||||
|     this.gson = gson; |     this.gson = gson; | ||||||
|  |     this.pauseUploads = new HashMap<>(); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   /** |   /** | ||||||
|  | @ -64,32 +70,50 @@ public class UploadClient { | ||||||
|   Observable<StashUploadResult> uploadFileToStash( |   Observable<StashUploadResult> uploadFileToStash( | ||||||
|       final Context context, final String filename, final Contribution contribution, |       final Context context, final String filename, final Contribution contribution, | ||||||
|       final NotificationUpdateProgressListener notificationUpdater) throws IOException { |       final NotificationUpdateProgressListener notificationUpdater) throws IOException { | ||||||
|     if (contribution.getChunkInfo() != null && contribution.getChunkInfo().isLastChunkUploaded()) { |     if (contribution.getChunkInfo() != null | ||||||
|  |         && contribution.getChunkInfo().getTotalChunks() == contribution.getChunkInfo() | ||||||
|  |         .getIndexOfNextChunkToUpload()) { | ||||||
|       return Observable.just(new StashUploadResult(StashUploadState.SUCCESS, |       return Observable.just(new StashUploadResult(StashUploadState.SUCCESS, | ||||||
|           contribution.getChunkInfo().getUploadResult().getFilekey())); |           contribution.getChunkInfo().getUploadResult().getFilekey())); | ||||||
|     } |     } | ||||||
|     pauseUploads = false; | 
 | ||||||
|     File file = new File(contribution.getLocalUri().getPath()); |     pauseUploads.put(contribution.getPageId(), false); | ||||||
|     final Observable<File> fileChunks = fileUtilsWrapper.getFileChunks(context, file, CHUNK_SIZE); | 
 | ||||||
|  |     final File file = new File(contribution.getLocalUri().getPath()); | ||||||
|  |     final List<File> fileChunks = fileUtilsWrapper.getFileChunks(context, file, CHUNK_SIZE); | ||||||
|  | 
 | ||||||
|  |     final int totalChunks = fileChunks.size(); | ||||||
|  | 
 | ||||||
|     final MediaType mediaType = MediaType |     final MediaType mediaType = MediaType | ||||||
|         .parse(FileUtils.getMimeType(context, Uri.parse(file.getPath()))); |         .parse(FileUtils.getMimeType(context, Uri.parse(file.getPath()))); | ||||||
| 
 | 
 | ||||||
|     final AtomicInteger index = new AtomicInteger(); |  | ||||||
|     final AtomicReference<ChunkInfo> chunkInfo = new AtomicReference<>(); |     final AtomicReference<ChunkInfo> chunkInfo = new AtomicReference<>(); | ||||||
|     Timber.d("Chunk info"); |     if (isStashValid(contribution)) { | ||||||
|     if (contribution.getChunkInfo() != null && isStashValid(contribution)) { |  | ||||||
|       chunkInfo.set(contribution.getChunkInfo()); |       chunkInfo.set(contribution.getChunkInfo()); | ||||||
|  | 
 | ||||||
|  |       Timber.d("Chunk: Next Chunk: %s, Total Chunks: %s", | ||||||
|  |           contribution.getChunkInfo().getIndexOfNextChunkToUpload(), | ||||||
|  |           contribution.getChunkInfo().getTotalChunks()); | ||||||
|     } |     } | ||||||
|     compositeDisposable.add(fileChunks.forEach(chunkFile -> { | 
 | ||||||
|       if (pauseUploads) { |     final AtomicInteger index = new AtomicInteger(); | ||||||
|  |     final AtomicBoolean failures = new AtomicBoolean(); | ||||||
|  | 
 | ||||||
|  |     compositeDisposable.add(Observable.fromIterable(fileChunks).forEach(chunkFile -> { | ||||||
|  |       if (pauseUploads.get(contribution.getPageId()) || failures.get()) { | ||||||
|         return; |         return; | ||||||
|       } |       } | ||||||
|       if (chunkInfo.get() != null && index.get() < chunkInfo.get().getLastChunkIndex()) { | 
 | ||||||
|         index.getAndIncrement(); |       if (chunkInfo.get() != null && index.get() < chunkInfo.get().getIndexOfNextChunkToUpload()) { | ||||||
|  |         index.incrementAndGet(); | ||||||
|  |         Timber.d("Chunk: Increment and return: %s", index.get()); | ||||||
|         return; |         return; | ||||||
|       } |       } | ||||||
|  |       index.getAndIncrement(); | ||||||
|       final int offset = |       final int offset = | ||||||
|           chunkInfo.get() != null ? chunkInfo.get().getUploadResult().getOffset() : 0; |           chunkInfo.get() != null ? chunkInfo.get().getUploadResult().getOffset() : 0; | ||||||
|  | 
 | ||||||
|  |       Timber.d("Chunk: Sending Chunk number: %s, offset: %s", index.get(), offset); | ||||||
|       final String filekey = |       final String filekey = | ||||||
|           chunkInfo.get() != null ? chunkInfo.get().getUploadResult().getFilekey() : null; |           chunkInfo.get() != null ? chunkInfo.get().getUploadResult().getFilekey() : null; | ||||||
| 
 | 
 | ||||||
|  | @ -104,17 +128,20 @@ public class UploadClient { | ||||||
|           offset, |           offset, | ||||||
|           filekey, |           filekey, | ||||||
|           countingRequestBody).subscribe(uploadResult -> { |           countingRequestBody).subscribe(uploadResult -> { | ||||||
|         chunkInfo.set(new ChunkInfo(uploadResult, index.incrementAndGet(), false)); |         Timber.d("Chunk: Received Chunk number: %s, offset: %s", index.get(), | ||||||
|  |             uploadResult.getOffset()); | ||||||
|  |         chunkInfo.set( | ||||||
|  |             new ChunkInfo(uploadResult, index.get(), totalChunks)); | ||||||
|         notificationUpdater.onChunkUploaded(contribution, chunkInfo.get()); |         notificationUpdater.onChunkUploaded(contribution, chunkInfo.get()); | ||||||
|       }, throwable -> { |       }, throwable -> { | ||||||
|         Timber.e(throwable, "Error occurred in uploading chunk"); |         failures.set(true); | ||||||
|       })); |       })); | ||||||
|     })); |     })); | ||||||
| 
 | 
 | ||||||
|     chunkInfo.get().setLastChunkUploaded(true); |     if (pauseUploads.get(contribution.getPageId())) { | ||||||
|     notificationUpdater.onChunkUploaded(contribution, chunkInfo.get()); |  | ||||||
|     if (pauseUploads) { |  | ||||||
|       return Observable.just(new StashUploadResult(StashUploadState.PAUSED, null)); |       return Observable.just(new StashUploadResult(StashUploadState.PAUSED, null)); | ||||||
|  |     } else if (failures.get()) { | ||||||
|  |       return Observable.just(new StashUploadResult(StashUploadState.FAILED, null)); | ||||||
|     } else if (chunkInfo.get() != null) { |     } else if (chunkInfo.get() != null) { | ||||||
|       return Observable.just(new StashUploadResult(StashUploadState.SUCCESS, |       return Observable.just(new StashUploadResult(StashUploadState.SUCCESS, | ||||||
|           chunkInfo.get().getUploadResult().getFilekey())); |           chunkInfo.get().getUploadResult().getFilekey())); | ||||||
|  | @ -129,8 +156,9 @@ public class UploadClient { | ||||||
|    * @return |    * @return | ||||||
|    */ |    */ | ||||||
|   private boolean isStashValid(Contribution contribution) { |   private boolean isStashValid(Contribution contribution) { | ||||||
|     return contribution.getDateModified() |     return contribution.getChunkInfo() != null && | ||||||
|         .after(new Date(System.currentTimeMillis() - MAX_CHUNK_AGE)); |         contribution.getDateModified() | ||||||
|  |             .after(new Date(System.currentTimeMillis() - MAX_CHUNK_AGE)); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   /** |   /** | ||||||
|  | @ -166,9 +194,10 @@ public class UploadClient { | ||||||
| 
 | 
 | ||||||
|   /** |   /** | ||||||
|    * Dispose the active disposable and sets the pause variable |    * Dispose the active disposable and sets the pause variable | ||||||
|  |    * @param pageId | ||||||
|    */ |    */ | ||||||
|   public void pauseUpload() { |   public void pauseUpload(String pageId) { | ||||||
|     pauseUploads = true; |     pauseUploads.put(pageId, true); | ||||||
|     if (!compositeDisposable.isDisposed()) { |     if (!compositeDisposable.isDisposed()) { | ||||||
|       compositeDisposable.dispose(); |       compositeDisposable.dispose(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -148,7 +148,7 @@ public class UploadService extends CommonsDaggerService { | ||||||
|    * @param contribution |    * @param contribution | ||||||
|    */ |    */ | ||||||
|   public void pauseUpload(Contribution contribution) { |   public void pauseUpload(Contribution contribution) { | ||||||
|     uploadClient.pauseUpload(); |     uploadClient.pauseUpload(contribution.getPageId()); | ||||||
|     contribution.setState(Contribution.STATE_PAUSED); |     contribution.setState(Contribution.STATE_PAUSED); | ||||||
|     compositeDisposable.add(contributionDao.update(contribution) |     compositeDisposable.add(contributionDao.update(contribution) | ||||||
|         .subscribeOn(ioThreadScheduler) |         .subscribeOn(ioThreadScheduler) | ||||||
|  | @ -312,8 +312,6 @@ public class UploadService extends CommonsDaggerService { | ||||||
|         .flatMap(uploadStash -> { |         .flatMap(uploadStash -> { | ||||||
|           notificationManager.cancel(notificationTag, NOTIFICATION_UPLOAD_IN_PROGRESS); |           notificationManager.cancel(notificationTag, NOTIFICATION_UPLOAD_IN_PROGRESS); | ||||||
| 
 | 
 | ||||||
|           Timber.d("Stash upload response 1 is %s", uploadStash.toString()); |  | ||||||
| 
 |  | ||||||
|           if (uploadStash.getState() == StashUploadState.SUCCESS) { |           if (uploadStash.getState() == StashUploadState.SUCCESS) { | ||||||
|             Timber.d("making sure of uniqueness of name: %s", filename); |             Timber.d("making sure of uniqueness of name: %s", filename); | ||||||
|             String uniqueFilename = findUniqueFilename(filename); |             String uniqueFilename = findUniqueFilename(filename); | ||||||
|  | @ -332,7 +330,6 @@ public class UploadService extends CommonsDaggerService { | ||||||
|               } |               } | ||||||
|             }); |             }); | ||||||
|           } else if (uploadStash.getState() == StashUploadState.PAUSED) { |           } else if (uploadStash.getState() == StashUploadState.PAUSED) { | ||||||
|             Timber.d("Contribution upload paused"); |  | ||||||
|             showPausedNotification(contribution); |             showPausedNotification(contribution); | ||||||
|             return Observable.never(); |             return Observable.never(); | ||||||
|           } else { |           } else { | ||||||
|  | @ -359,7 +356,6 @@ public class UploadService extends CommonsDaggerService { | ||||||
| 
 | 
 | ||||||
|   private void onUpload(Contribution contribution, String notificationTag, |   private void onUpload(Contribution contribution, String notificationTag, | ||||||
|       UploadResult uploadResult) { |       UploadResult uploadResult) { | ||||||
|     Timber.d("Stash upload response 2 is %s", uploadResult.toString()); |  | ||||||
| 
 | 
 | ||||||
|     notificationManager.cancel(notificationTag, NOTIFICATION_UPLOAD_IN_PROGRESS); |     notificationManager.cancel(notificationTag, NOTIFICATION_UPLOAD_IN_PROGRESS); | ||||||
| 
 | 
 | ||||||
|  | @ -401,7 +397,7 @@ public class UploadService extends CommonsDaggerService { | ||||||
| 
 | 
 | ||||||
|   @SuppressLint("StringFormatInvalid") |   @SuppressLint("StringFormatInvalid") | ||||||
|   @SuppressWarnings("deprecation") |   @SuppressWarnings("deprecation") | ||||||
|   private void showFailedNotification(Contribution contribution) { |   private void showFailedNotification(final Contribution contribution) { | ||||||
|     final String displayTitle = contribution.getMedia().getDisplayTitle(); |     final String displayTitle = contribution.getMedia().getDisplayTitle(); | ||||||
|     curNotification.setTicker(getString(R.string.upload_failed_notification_title, displayTitle)) |     curNotification.setTicker(getString(R.string.upload_failed_notification_title, displayTitle)) | ||||||
|         .setContentTitle(getString(R.string.upload_failed_notification_title, displayTitle)) |         .setContentTitle(getString(R.string.upload_failed_notification_title, displayTitle)) | ||||||
|  | @ -412,6 +408,7 @@ public class UploadService extends CommonsDaggerService { | ||||||
|         curNotification.build()); |         curNotification.build()); | ||||||
| 
 | 
 | ||||||
|     contribution.setState(Contribution.STATE_FAILED); |     contribution.setState(Contribution.STATE_FAILED); | ||||||
|  |     contribution.setChunkInfo(null); | ||||||
| 
 | 
 | ||||||
|     compositeDisposable.add(contributionDao |     compositeDisposable.add(contributionDao | ||||||
|         .update(contribution) |         .update(contribution) | ||||||
|  | @ -419,7 +416,7 @@ public class UploadService extends CommonsDaggerService { | ||||||
|         .subscribe()); |         .subscribe()); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   private void showPausedNotification(Contribution contribution) { |   private void showPausedNotification(final Contribution contribution) { | ||||||
|     final String displayTitle = contribution.getMedia().getDisplayTitle(); |     final String displayTitle = contribution.getMedia().getDisplayTitle(); | ||||||
|     curNotification.setTicker(getString(R.string.upload_paused_notification_title, displayTitle)) |     curNotification.setTicker(getString(R.string.upload_paused_notification_title, displayTitle)) | ||||||
|         .setContentTitle(getString(R.string.upload_paused_notification_title, displayTitle)) |         .setContentTitle(getString(R.string.upload_paused_notification_title, displayTitle)) | ||||||
|  |  | ||||||
|  | @ -7,7 +7,7 @@ buildscript { | ||||||
|         maven { url "https://plugins.gradle.org/m2/" } |         maven { url "https://plugins.gradle.org/m2/" } | ||||||
|     } |     } | ||||||
|     dependencies { |     dependencies { | ||||||
|         classpath 'com.android.tools.build:gradle:4.0.0' |         classpath 'com.android.tools.build:gradle:4.0.1' | ||||||
|         classpath "com.hiya:jacoco-android:0.2" |         classpath "com.hiya:jacoco-android:0.2" | ||||||
|         classpath 'com.getkeepsafe.dexcount:dexcount-gradle-plugin:0.8.2' |         classpath 'com.getkeepsafe.dexcount:dexcount-gradle-plugin:0.8.2' | ||||||
|         classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$KOTLIN_VERSION" |         classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$KOTLIN_VERSION" | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Vivek Maskara
						Vivek Maskara