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:
Vivek Maskara 2020-10-24 23:56:48 -07:00 committed by GitHub
parent 0d5fa048a5
commit 6c55525a43
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
8 changed files with 69 additions and 38 deletions

View file

@ -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())

View file

@ -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 {

View file

@ -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

View file

@ -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)

View file

@ -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;
} }
} }

View file

@ -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;
}
if (chunkInfo.get() != null && index.get() < chunkInfo.get().getIndexOfNextChunkToUpload()) {
index.incrementAndGet();
Timber.d("Chunk: Increment and return: %s", index.get());
return; return;
} }
if (chunkInfo.get() != null && index.get() < chunkInfo.get().getLastChunkIndex()) {
index.getAndIncrement(); index.getAndIncrement();
return;
}
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,7 +156,8 @@ public class UploadClient {
* @return * @return
*/ */
private boolean isStashValid(Contribution contribution) { private boolean isStashValid(Contribution contribution) {
return contribution.getDateModified() return contribution.getChunkInfo() != null &&
contribution.getDateModified()
.after(new Date(System.currentTimeMillis() - MAX_CHUNK_AGE)); .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();
} }

View file

@ -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))

View file

@ -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"