Проблема с объединением веб-клиента с Mono и Flux - PullRequest
0 голосов
/ 03 мая 2019

Привет, у меня есть Flux, и во время итерации по каждому элементу создается новый моно.У меня также есть другие моно вне потока.И хотите сделать следующее: Когда флюс (с соответствующим внутренним концом моно), тогда второй моно.Практика заключается в том, что поток внутри моно создается из запроса веб-клиента.В качестве отправной точки, пожалуйста, обратите внимание на метод загрузки.В основном без веб-клиента это работает, но в случае с веб-клиентом внутри карты работа после этого.Использование пружинной загрузки 2

public WebClient.ResponseSpec sendGetRequest(String path, Object... pathVariables){
   try {
       LOGGER.info("content type {}, url {}, path {}", contentType, url, path);
       WebClient.ResponseSpec responseSpec = sendRequest(HttpMethod.GET, contentType, authorizationToken, url, path, pathVariables);
       return responseSpec;
   }catch (Exception e){
       throw new WebClientProcessingException("Exception when trying to process", e);
   }
}

public Mono<PersonPayload> loadPerson(String  path){
    try {
        LOGGER.info("path {}", path);
        Mono<QuestionDetailsPayload> person = sendGetRequest(path).bodyToMono(PersonPayload.class);
        return person;
    }catch (Exception e){
        throw new WebClientProcessingException("Exception when trying to process",e);
    }
}


public Mono<PersonDomain> getPerson(String path) {
    Assert.notNull(path, "path can't be null");
    try{
        LOGGER.info("path {}" ,path);
        Mono<PersonPayload> personPayload = loadPerson(path);
        return personPayload.map(this::toPersonDomain);
    }catch (Exception e){
        throw new PersonNotFoundException("Exception when trying to get person info" , e);

    }
}

public PersponDomain toPersonDomain(PersonPayload personPayload){
    return modelMapper.map(personPayload, PersonDomain.class);
}

public void load(){
    List<String> outStr = Arrays.asList("out1", "out2","out3");
    Flux flux = Flux.fromIterable(outStr);
    Flux<Mono<PersonDomain>> results =  flux.map(string ->{
        System.out.println(string);
        Mono<PersonDomain> personMono = getPerson("inside");
        Mono<String> result = personMono.map(h ->{
            System.out.println(personMono.getName());
            return personMono.getName() + "_test";
        });
        return result;
    });
    Mono<String> second = Mono.just("second");
    results.then(second);
    results.subscribe(stringMono -> {
        stringMono.subscribe();
    });
    second.subscribe( s->{
        System.out.println(s);
    });
}

Зависимость Gradle:

implementation 'org.springframework.boot:spring-boot-starter-webflux'
implementation 'org.springframework.boot:spring-boot-starter-validation'

implementation 'org.postgresql:postgresql'
implementation 'org.springframework.boot:spring-boot-starter-jooq'
implementation 'org.jooq:jooq-codegen'

implementation 'org.modelmapper:modelmapper:2.3.0'
implementation 'org.modelmapper:modelmapper:2.3.0'
compileOnly 'org.projectlombok:lombok'
annotationProcessor 'org.projectlombok:lombok'

implementation 'com.google.code.gson:gson:2.8.5'

testImplementation 'org.springframework.boot:spring-boot-starter-test'
testImplementation 'io.projectreactor:reactor-test'
testImplementation 'org.powermock:powermock-module-junit4:2.0.0'
testImplementation 'org.powermock:powermock-api-mockito2:2.0.0'}

Ответы [ 2 ]

0 голосов
/ 08 мая 2019

Решение для моего случая использовать:

subscribe(Consumer<? super T> consumer,
          Consumer<? super Throwable> errorConsumer,
          Runnable completeConsumer); 

вместо того. Которые имеют дело со значениями и ошибками, но также выполняют некоторый код после успешного завершения последовательности.

0 голосов
/ 03 мая 2019

Flux#map является синхронной операцией и не подписывается на возвращаемый объект.

Вы должны использовать Flux#flatMap / Flux#concatMap / Flux#flatMapSequential / Flux#switchMap. Эти операторы подпишутся на возвращенные Publisher.

...