В настоящее время я работаю над частью приложения, где пользователю нужно выбрать несколько изображений из галереи, а затем загрузить эти изображения на сервер. для загрузки я использую модификацию 2.0, используя multipart.
Следуя за ресурсом https://futurestud.io/blog/retrofit-2-how-to-upload-files-to-server Мне удалось успешно реализовать его при использовании только одного файла, однако мне бы хотелось, чтобы за один сеанс все изображения, которые пользователь выбрал для загрузки на сервер.
Оглядываясь вокруг я нашел "модифицировать" несколько изображений, прикрепленных в одном запросе где это обсуждается в аннотации @partmap, и это похоже на правильное совпадение, однако я не понимаю, как перебирать изображения, которые мне нужно загрузить?
может кто-нибудь указать мне на правильную реализацию?
Также проверено:
1.Модификация (2.0 бета2) Многочастная загрузка файлов не работает
2.Загрузить несколько изображений с тем же именем, что и массив Retrofit 2.0.0-beta2
Я просто пытаюсь решить аналогичный случай, но у меня есть большое количество файлов.
Я попытался решить это с помощью цикла:
recordsDirectory.mkdirs();
final File[] list_files = recordsDirectory.listFiles();
int cont = list_files.length;
for (int i = 0; i < cont; i++) {
final File ref_file = list_files[i];
String source_filepath = ref_file.getPath();
RequestBody requestBody =
RequestBody.create(MediaType.parse("multipart/form-data"), ref_file);
Call<DeliveryResponse> call = apiService.upload(requestBody, description);
call.enqueue(new Callback<DeliveryResponse>() {
@Override
public void onResponse(Response<DeliveryResponse> response, Retrofit retrofit) {
Log.v(TAG, "[" + TAG + "] Upload success. status: " + response.toString());
DeliveryResponse deliveryResponse = response.body();
Log.v(TAG, "response code: " + deliveryResponse.mStatus);
Log.v(TAG, "response code: " + response.code());
Log.v(TAG, "ref_file: " + ref_file.getPath());
if (response.isSuccess()) {
Log.i(TAG, "[" + TAG + "] The server has response with a status distinct to 200.");
}
else {
if (deliveryResponse.mStatus == 1) {
Log.i(TAG, "[" + TAG + "] The server has response with 1 as status.");
boolean deleted = ref_file.delete();
} else {
Log.i(TAG, "[" + TAG + "] The server has response with 0 as status.");
}
}
}
@Override
public void onFailure(Throwable t) {
Log.e("Upload", t.getMessage());
}
});
}
/***********************************************
RETROFIT API
************************************************/
public interface MyApiEndpointInterface {
@Multipart
@POST("/elderbask/api/sensors/accelerometer")
Call<DeliveryResponse> upload(
@Part("myfile\"; filename=\"image.png\" ") RequestBody file,
@Part("description") String description
);
}
public static class DeliveryResponse {
@SerializedName("status")
int mStatus;
public DeliveryResponse(int status) {
this.mStatus = status;
}
public String toString(){
return "mStatus: " + this.mStatus;
}
}
Но это решение работает не так хорошо, как я ожидаю. У меня есть много исключений тайм-аута, и неверно закодированное имя файла довольно ужасно.
Так как мне не удалось решить проблему с помощью модернизации, я использовал библиотеку okhttp, которая дала мне то, что мне было нужно, вот пример кода, который я использую, который, я надеюсь, может кому-то помочь
построить тело запроса:
okhttp3.RequestBody requestBody = new MultipartBody.Builder()
.setType(MultipartBody.FORM)
.addPart(okhttp3.Headers.of("Content-Disposition:", "form-data; name=\"myfile\"; filename=\"image" + ImageNumber + ".jpg\""),
okhttp3.RequestBody.create(MEDIA_TYPE_JPG, new File(ImagePath)))
.build();
подготовить запрос:
request = new okhttp3.Request.Builder()
.header("Authorization", "Client-ID " )
.url(YOUR URL)
.post(requestBody)
.build();
инициировать соединение с сервером:
try {
okhttp3.Response response = client.newCall(request).execute();
if (!response.isSuccessful())
throw new IOException("Unexpected code " + response);
System.out.println(response.body().string());
} catch (IOException e) {
e.printStackTrace();
}
Wooh! Я сделал, вы можете попробовать это, как я (Retrofit 2)
//1. What We Need From Server ( upload.php Script )
public class FromServer {
String result;
}
//2. Which Interface To Communicate Our upload.php Script?
public interface ServerAPI {
@Multipart
@POST("upload.php")//Our Destination PHP Script
Call<List<FromServer>> upload(
@Part("file_name") String file_name,
@Part("file") RequestBody file);
Retrofit retrofit =
new Retrofit.Builder()
.baseUrl("http://192.168.43.135/retro/") // REMEMBER TO END with /
.addConverterFactory(GsonConverterFactory.create())
.build();
}//3. How To Upload
private void upload(){
ServerAPI api = ServerAPI.retrofit.create(ServerAPI.class);
File from_phone = FileUtils.getFile(Environment.getExternalStorageDirectory()+"/aa.jpg"); //org.apache.commons.io.FileUtils
RequestBody to_server = RequestBody.create(MediaType.parse("multipart/form-data"), from_phone);
api.upload(from_phone.getName(),to_server).enqueue(new Callback<List<FromServer>>() {
@Override
public void onResponse(Call<List<FromServer>> call, Response<List<FromServer>> response) {
Toast.makeText(MainActivity.this, response.body().get(0).result, Toast.LENGTH_SHORT).show();
}
@Override
public void onFailure(Call<List<FromServer>> call, Throwable t) { }
});}
//4. upload.php
<?php
$pic = $_POST['file_name'];
$pic = str_replace("\"", "", $pic); //REMOVE " from file name
$f = fopen($pic, "w");
fwrite($f,$_POST['file']);
fclose($f);
$arr[] = array("result"=>"Done");
print(json_encode($arr));
?>