Я использую Retrofit2 для обработки вызовов API в моем приложении Android. Приложение содержит несколько функций загрузки изображений. Все функции загрузки изображений подключены до двух вызовов API и будут запускаться один за другим.
Первый API будет загружать содержимое изображения на сервер, а сервер будет генерировать динамический подписанный URL-адрес изображения AWS S3 в качестве ответа.
Второй API использует подписанный выше URL и помещает данные в AWS S3.
Это работало отлично для маленьких изображений. Для больших изображений API не удалось. Возможное решение этой проблемы - изменить API загрузки как составной. Поэтому я изменил второй API как " Retrofit2-multipart ". Но проблема здесь в том, что после загрузки изображения файл поврежден в S3.
Итак, мой вопрос: подходит ли модификация multipart для загрузки ASW S3? Есть ли у нас какое-либо решение для решения этой проблемы?
Пожалуйста, проверьте мою реализацию
Модифицированный интерфейс API
@Multipart
@PUT
Call<Void> uploadFile(@Url String url,
@Part MultipartBody.Part file);
Создатель модифицированного сервиса
private ApiServicesList createService() {
OkHttpClient okHttpClient = new OkHttpClient.Builder()
.readTimeout(300, TimeUnit.SECONDS)
.connectTimeout(360, TimeUnit.SECONDS)
.cache(null)
.addInterceptor(new NetworkInterceptor(context))
.addInterceptor(createLoggingInterceptor())
.addInterceptor(createSessionExpiryInterceptor())
.addInterceptor(createContextHeaderInterceptor())
.build();
return new Retrofit.Builder()
.baseUrl(FirebaseConfig.getInstance().getStagingBaseURl())
.addConverterFactory(GsonConverterFactory.create())
.client(okHttpClient)
.build()
.create(ApiServicesList.class);
}
Класс загрузки файлов
public void uploadFileTos3(String url, MultipartBody.Part file, Callback<Void> listener) {
Call<Void> call = mAPIServices.uploadFile(url,file);
call.enqueue(listener);
}
Вызов API Presenter
networkServices.uploadFileTos3(credentials.getSignedUrl(), prepareRequestBody(file), new Callback<Void>() {
@Override
public void onFailure(Call<Void> call, Throwable t) {
//App Action
}
@Override
public void
onResponse(Call<Void> call, Response<Void> response) {
//App Action
}
});
Метод тела запроса
public static MultipartBody.Part prepareRequestBody(File file) {
RequestBody requestFile = RequestBody.create(MediaType.parse(CONTENT_TYPE), file); // Checked Both "image/jpeg" and "multipart/form-data"
return MultipartBody.Part.createFormData("image", file.getName(), requestFile);
}
DDMS LOGS
2019-04-10 19:07:13.926 18761-18874/com.xxx D/OkHttp: --> PUT https:Signed URL(Removed actual URL)
2019-04-10 19:07:13.926 18761-18874/com.xxx D/OkHttp: Content-Type: multipart/form-data; boundary=60561d1c-ff3f-4a43-8022-ca2be3e8ec4e
2019-04-10 19:07:13.926 18761-18874/com.xxx D/OkHttp: Content-Length: 71154
2019-04-10 19:07:13.931 18761-18874/com.xxx D/OkHttp: --60561d1c-ff3f-4a43-8022-ca2be3e8ec4e
2019-04-10 19:07:13.931 18761-18874/com.xxx D/OkHttp: Content-Disposition: form-data; name="image"; filename="forest-trees-fog-foggy.jpg"
2019-04-10 19:07:13.931 18761-18874/com.xxx D/OkHttp: Content-Type: image/jpeg
2019-04-10 19:07:13.931 18761-18874/com.xxx D/OkHttp: Content-Length: 70934
2019-04-10 19:07:13.931 18761-18874/com.xxx D/OkHttp: Image Body
2019-04-10 19:07:13.942 18761-18874/com.xxx D/OkHttp: --60561d1c-ff3f-4a43-8022-ca2be3e8ec4e--
2019-04-10 19:07:13.942 18761-18874/com.xxx D/OkHttp: --> END PUT (71154-byte body)
2019-04-10 19:07:15.648 18761-18874/com.xxx D/OkHttp: <-- 200 OK
Попробуйте изменить вызов createFormData с помощью "файла" вместо "изображения":
public static MultipartBody.Part prepareRequestBody(File file) {
RequestBody requestFile = RequestBody.create(MediaType.parse(CONTENT_TYPE), file); // Checked Both "image/jpeg" and "multipart/form-data"
return MultipartBody.Part.createFormData("file", file.getName(), requestFile);
}