如何使用OkHTTP3 / Retrofit2获取多部分多文件上传的进度?

时间:2017-07-10 18:38:14

标签: android retrofit2 okhttp multipart

我使用Retrofit2在单个多部分请求中上传动态数量的文件。我的改造界面如下所示 -

public interface FileUploadService {  
    @Multipart
    @POST("upload")
    Call<ResponseBody> uploadMultipleFilesDynamic(
            @Part List<MultipartBody.Part> files);
}

现在我想跟踪这个多文件上传的进度。 This solution解释了如何通过扩展RequestBody在多部分请求中上传单个文件时取得进展。虽然我似乎无法理解如何将此应用于我的多个文件请求。我能想到的一个解决方案是通过扩展ProgressRequestBody类而不是OkHTTP MultipartBody创建RequestBody,但OkHTTP3实现MultipartBody作为最终类,因此无法扩展。任何人都可以指出我正确的方向,因为它是一个巨大的阻止我无法向用户显示文件上传的进度。或者,我是否可以实现任何可以实现此功能的工作?

1 个答案:

答案 0 :(得分:0)

我已按照此博文:https://medium.com/@PaulinaSadowska/display-progress-of-multipart-request-with-retrofit-and-rxjava-23a4a779e6ba进行了以下调整,以显示总进度而非单独文件的进展情况:

  private fun prepareFileParts(reportAttachments: MutableList<ReportAttachment>, emitter: FlowableEmitter<Double>): List<MultipartBody.Part> {
    val multiPartBodyList = mutableListOf<MultipartBody.Part>()

    var offset = 0L
    var totalLength = 0L

    // calculate the total length of all files
    for (attachment in reportAttachments) {
        val file = File(attachment.filePath)
        totalLength += file.length()
    }

    // create requestbody for each file and calculate the progression offset
    for (attachment in reportAttachments) {
        val file = File(attachment.filePath)
        val mimeType = attachment.mimeType

        val countingRequestBody = createCountingRequestBody(file, mimeType, emitter, offset, totalLength)
        offset += file.length()

        val multipartBody = MultipartBody.Part.createFormData("file", file.name, countingRequestBody)
        multiPartBodyList.add(multipartBody)
    }

    return multiPartBodyList
}

private fun createCountingRequestBody(file: File, mimeType: String, emitter: FlowableEmitter<Double>, offset: Long, totalLength: Long): RequestBody {
    val requestBody = RequestBody.create(MediaType.parse(mimeType), file)
    return CountingRequestBody(requestBody, object : CountingRequestBody.Listener {
        override fun onRequestProgress(bytesWritten: Long, contentLength: Long) {
            val progress: Double = 1.0 * (offset + bytesWritten) / totalLength
            emitter.onNext(progress)
        }
    })
}

如果需要,您还可以创建拦截器并将其添加到OkHttpClient。这将默认跟踪所有传出的API调用。它看起来像这样:

class UploadProgressInterceptor(private val progressListener: CountingRequestBody.Listener) : Interceptor {

override fun intercept(chain: Interceptor.Chain): Response {
    val originalRequest = chain.request()

    if (originalRequest.body() == null) {
        return chain.proceed(originalRequest)
    }

    val requestBody = originalRequest.body()
    requestBody?.let {
        val progressRequest = originalRequest.newBuilder()
                .method(originalRequest.method(), CountingRequestBody(it, progressListener))
                .build()
        return chain.proceed(progressRequest)
    }
    return chain.proceed(originalRequest)
}