RxAndroid и Retrofit: невозможно создать адаптер вызова для io.reactivex.Observable <retrofit2.Response <okhttp3.ResponseBody >>

Я пытаюсь использовать rxJava, rxAndroid, Retrofit2 и OkHTTP3 для загрузки файла с конечной точки URL. Мой код не может создать адаптер вызова для "Observable < retrofit2.Response < okhttp3.ResponseBody → ". Эти методы новы для меня, поэтому я считаю, что здесь отсутствует важная концепция. Любое направление или точки приветствуются.

FATAL EXCEPTION: главная Процесс: файл com.example.khe11e.rxdownload, PID: 14130 java.lang.IllegalArgumentException: невозможно создать адаптер вызова для io.reactivex.Observable >    для метода RetrofitInterface.downloadFileByUrlRx    в retrofit2.ServiceMethod $Builder.methodError(ServiceMethod.java:720)    в retrofit2.ServiceMethod $Builder.createCallAdapter(ServiceMethod.java:234)    в retrofit2.ServiceMethod $Builder.build(ServiceMethod.java:160)    на retrofit2.Retrofit.loadServiceMethod(Retrofit.java:166)    в retrofit2.Retrofit $1.invoke(Retrofit.java:145)    в java.lang.reflect.Proxy.invoke(Proxy.java:393)    в $Proxy0.downloadFileByUrlRx(Неизвестный источник)    at com.example.khe11e.rxdownloadfile.MainActivity.downloadImage(MainActivity.java:46)    at com.example.khe11e.rxdownloadfile.MainActivity $1.onClick(MainActivity.java:39)    at android.view.View.performClick(View.java:5207)    на android.view.View $PerformClick.run(View.java:21168)    на android.os.Handler.handleCallback(Handler.java:746)    на android.os.Handler.dispatchMessage(Handler.java:95)    на android.os.Looper.loop(Looper.java:148)    at android.app.ActivityThread.main(ActivityThread.java:5491)    в java.lang.reflect.Method.invoke(собственный метод)    at com.android.internal.os.ZygoteInit $MethodAndArgsCaller.run(ZygoteInit.java:728)    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:618) Вызвано: java.lang.IllegalArgumentException: Не удалось найти адаптер вызова для io.reactivex.Observable > .  Пытался:   * retrofit2.adapter.rxjava.RxJavaCallAdapterFactory   * retrofit2.ExecutorCallAdapterFactory    на retrofit2.Retrofit.nextCallAdapter(Retrofit.java:237)    на retrofit2.Retrofit.callAdapter(Retrofit.java:201)    в retrofit2.ServiceMethod $Builder.createCallAdapter(ServiceMethod.java:232)     ... еще 16

build.gradle:

compile 'io.reactivex.rxjava2:rxandroid:2.0.1'
compile 'io.reactivex.rxjava2:rxjava:2.0.4'
compile 'com.squareup.retrofit2:retrofit:2.1.0'
compile 'com.squareup.retrofit2:adapter-rxjava:2.1.0'

RetrofitInterface.java:

package com.example.khe11e.rxdownloadfile;
import io.reactivex.Observable;
import okhttp3.ResponseBody;
import retrofit2.Call;
import retrofit2.Response;
import retrofit2.http.GET;
import retrofit2.http.Streaming;
import retrofit2.http.Url;

public interface RetrofitInterface {
    // Retrofit 2 GET request for rxjava
    @Streaming
    @GET
    Observable<Response<ResponseBody>> downloadFileByUrlRx(@Url String fileUrl);
}

MainActivity.java:

package com.example.khe11e.rxdownloadfile;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import java.io.File;
import io.reactivex.Observable;
import io.reactivex.ObservableEmitter;
import io.reactivex.ObservableOnSubscribe;
import io.reactivex.Observer;
import io.reactivex.android.schedulers.AndroidSchedulers;
import io.reactivex.disposables.Disposable;
import io.reactivex.functions.Function;
import io.reactivex.schedulers.Schedulers;
import okhttp3.OkHttpClient;
import okhttp3.ResponseBody;
import okio.BufferedSink;
import okio.Okio;
import retrofit2.Response;
import retrofit2.Retrofit;
import retrofit2.adapter.rxjava.RxJavaCallAdapterFactory;

public class MainActivity extends AppCompatActivity {

Button downloadImgBtn;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    downloadImgBtn = (Button) findViewById(R.id.downloadImgBtn);
    downloadImgBtn.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            downloadImage();
        }
    });
}

public void downloadImage(){
    RetrofitInterface downloadService = createService(RetrofitInterface.class, "https://www.nasa.gov/");
    downloadService.downloadFileByUrlRx("sites/default/files/iss_1.jpg")
            .flatMap(processResponse())
            .subscribeOn(Schedulers.io())
            .observeOn(AndroidSchedulers.mainThread())
            .subscribe(handleResult());
}

public <T> T createService(Class<T> serviceClass, String baseUrl){
    Retrofit retrofit = new Retrofit.Builder()
            .baseUrl(baseUrl)
            .client(new OkHttpClient.Builder().build())
            .addCallAdapterFactory(RxJavaCallAdapterFactory.create()).build();
    return retrofit.create(serviceClass);
}

public Function<Response<ResponseBody>, Observable<File>> processResponse(){
    return new Function<Response<ResponseBody>, Observable<File>>() {
        @Override
        public Observable<File> apply(Response<ResponseBody> responseBodyResponse) throws Exception {
            return saveToDiskRx(responseBodyResponse);
        }
    };
}

private Observable<File> saveToDiskRx(final Response<ResponseBody> response){
    return Observable.create(new ObservableOnSubscribe<File>() {
        @Override
        public void subscribe(ObservableEmitter<File> subscriber) throws Exception {
            String header = response.headers().get("Content-Disposition");
            String filename = header.replace("attachment; filename=", "");
            new File("/data/data/" + getPackageName() + "/images").mkdir();
            File destinationFile = new File("/data/data/" + getPackageName() + "/images/" + filename);

            BufferedSink bufferedSink = Okio.buffer(Okio.sink(destinationFile));
            bufferedSink.writeAll(response.body().source());
            bufferedSink.close();

            subscriber.onNext(destinationFile);
            subscriber.onComplete();
        }
    });
}

private Observer<File> handleResult(){
    return new Observer<File>() {
        @Override
        public void onSubscribe(Disposable d) {
            Log.d("OnSubscribe", "OnSubscribe");
        }

        @Override
        public void onNext(File file) {
            Log.d("OnNext", "File downloaded to " + file.getAbsolutePath());
        }

        @Override
        public void onError(Throwable e) {
            e.printStackTrace();
            Log.d("Error", "Error " + e.getMessage());
        }

        @Override
        public void onComplete() {
            Log.d("OnComplete", "onCompleted");
        }
    };
}
}

Я попробовал добавить Call as упомянутый здесь, чтобы он выглядел так:

Call<Observable<Response<ResponseBody>>> downloadFileByUrlRx(@Url String fileUrl);

однако это вызывает проблемы с функцией flatMap, поскольку он не может найти метод символов flatMap (Function < Response <ResponseBody> , Observable < File → ).

Ответ 1

Вы используете адаптер RxJava 1 для переоснащения, замените его на вариант RxJava 2:

//compile 'com.squareup.retrofit2:adapter-rxjava:2.1.0'
compile 'com.jakewharton.retrofit:retrofit2-rxjava2-adapter:1.0.0'

UPDATE

Начиная с версии дооснащения 2.2.0 для RxJava2 существует адаптер для первой стороны:

compile 'com.squareup.retrofit2:retrofit:2.2.0'
compile 'com.squareup.retrofit2:adapter-rxjava2:2.2.0'

Ответ 2

Обновление: июнь /2019

Rxjava версия обновлена до 2.2.10

compile 'io.reactivex.rxjava2:rxandroid:2.1.1'
compile 'io.reactivex.rxjava2:rxjava:2.2.10'
compile 'com.squareup.retrofit2:retrofit:2.6.0'
compile 'com.squareup.retrofit2:adapter-rxjava2:2.6.0'

Также для модернизации требуется как минимум Java 7 или Android 2.3

------------------------------------------

Для новичков (июль/2017):

Вы перепутали свою версию библиотеки наверняка.

Я использую последнюю версию RXAndroid 2.0.1

compile 'io.reactivex.rxjava2:rxandroid:2.0.1'
compile 'io.reactivex.rxjava2:rxjava:2.1.0'
compile 'com.squareup.retrofit2:retrofit:2.3.0'
compile 'com.squareup.retrofit2:adapter-rxjava2:2.3.0'

1) Убедитесь, что у вас есть одинаковые адаптеры Retrofit и Retrofit-RxJava.

2) Используйте компиляцию

compile 'com.squareup.retrofit2:adapter-rxjava2:2.3.0' 

не

compile 'com.squareup.retrofit2:adapter-rxjava:2.3.0'

и используйте RxJava2CallAdapterFactory.create() вместо RxJavaCallAdapterFactory.create() при создании модернизации.

Ответ 3

Поскольку вы используете заголовок ответа, замените ResponseBody на Object всюду.