Spring 5でWebClientからのFluxをキャッシュする方法はありますか?私はこれを試しましたが、何もキャッシュしていません。
@RestController
@SpringBootApplication
@EnableCaching
public class GatewayApplication {
@PostMapping(value ="/test", produces = "application/json")
public Flux<String> handleRequest(@RequestBody String body) {
return getHspadQuery(body);
}
@Cacheable("testCache")
private Flux<String> getData (String body) {
return WebClient.create().post()
.uri("http://myurl")
.body(BodyInserters.fromObject(body))
.retrieve().bodyToFlux(String.class).cache();
}
}
私が3番目の要求をしたとき、それは決して終わらない。そして、その後のリクエストで応答を受け取りますが、サーバーは次をスローします。
2018-04-09 12:36:23.920 ERROR 11488 --- [ctor-http-nio-4] r.ipc.netty.channel.ChannelOperations : [HttpServer] Error processing connection. Requesting close the channel
reactor.core.Exceptions$OverflowException: Could not emit buffer due to lack of requests
at reactor.core.Exceptions.failWithOverflow(Exceptions.Java:215) ~[reactor-core-3.1.5.RELEASE.jar:3.1.5.RELEASE]
at reactor.core.publisher.FluxBufferPredicate$BufferPredicateSubscriber.emit(FluxBufferPredicate.Java:292) ~[reactor-core-3.1.5.RELEASE.jar:3.1.5.RELEASE]
at reactor.core.publisher.FluxBufferPredicate$BufferPredicateSubscriber.onNextNewBuffer(FluxBufferPredicate.Java:251) ~[reactor-core-3.1.5.RELEASE.jar:3.1.5.RELEASE]
at reactor.core.publisher.FluxBufferPredicate$BufferPredicateSubscriber.tryOnNext(FluxBufferPredicate.Java:205) ~[reactor-core-3.1.5.RELEASE.jar:3.1.5.RELEASE]
at reactor.core.publisher.FluxBufferPredicate$BufferPredicateSubscriber.onNext(FluxBufferPredicate.Java:180) ~[reactor-core-3.1.5.RELEASE.jar:3.1.5.RELEASE]
at reactor.core.publisher.FluxFlatMap$FlatMapMain.drainLoop(FluxFlatMap.Java:646) ~[reactor-core-3.1.5.RELEASE.jar:3.1.5.RELEASE]
at reactor.core.publisher.FluxFlatMap$FlatMapMain.drain(FluxFlatMap.Java:523) ~[reactor-core-3.1.5.RELEASE.jar:3.1.5.RELEASE]
at reactor.core.publisher.FluxFlatMap$FlatMapInner.onSubscribe(FluxFlatMap.Java:897) ~[reactor-core-3.1.5.RELEASE.jar:3.1.5.RELEASE]
at reactor.core.publisher.FluxIterable.subscribe(FluxIterable.Java:128) ~[reactor-core-3.1.5.RELEASE.jar:3.1.5.RELEASE]
at reactor.core.publisher.FluxIterable.subscribe(FluxIterable.Java:61) ~[reactor-core-3.1.5.RELEASE.jar:3.1.5.RELEASE]
at reactor.core.publisher.Flux.subscribe(Flux.Java:6873) ~[reactor-core-3.1.5.RELEASE.jar:3.1.5.RELEASE]
at reactor.core.publisher.FluxFlatMap$FlatMapMain.onNext(FluxFlatMap.Java:372) ~[reactor-core-3.1.5.RELEASE.jar:3.1.5.RELEASE]
at reactor.core.publisher.FluxMap$MapSubscriber.onNext(FluxMap.Java:108) ~[reactor-core-3.1.5.RELEASE.jar:3.1.5.RELEASE]
at reactor.core.publisher.FluxMap$MapSubscriber.onNext(FluxMap.Java:108) ~[reactor-core-3.1.5.RELEASE.jar:3.1.5.RELEASE]
at reactor.ipc.netty.channel.FluxReceive.drainReceiver(FluxReceive.Java:211) ~[reactor-netty-0.7.5.RELEASE.jar:0.7.5.RELEASE]
at reactor.ipc.netty.channel.FluxReceive.onInboundNext(FluxReceive.Java:326) ~[reactor-netty-0.7.5.RELEASE.jar:0.7.5.RELEASE]
...
そして、それは何もキャッシュしません。
どんな助けでもいただければ幸いです。
ありがとう。
最後にMonoで解決しました。たとえばreduceを使用するFluxで可能だと思います。
@RestController
@SpringBootApplication
public class Application {
@Autowired
CacheManager manager;
private WebClient client;
@PostConstruct
public void setup() {
client = WebClient.builder()
.baseUrl("http://myurl")
.exchangeStrategies(ExchangeStrategies.withDefaults())
.build();
}
@Bean
public CacheManager cacheManager() {
SimpleCacheManager cacheManager = new SimpleCacheManager();
cacheManager.setCaches(Arrays.asList(new ConcurrentMapCache("hspad")));
return cacheManager;
}
@PostMapping(value = "/hspad/grahql", produces = "application/json")
public Mono<ResponseEntity<String>> hspadService(@RequestBody String body) {
return getHspadQuery(body);
}
private Mono<ResponseEntity<String>> getHspadQuery (String body) {
Mono<ResponseEntity<String>> mono;
Optional<Cache.ValueWrapper> value = Optional.ofNullable(cacheManager().getCache("hspad").get(body));
if(value.isPresent()) {
mono = Mono.just(ResponseEntity.ok(value.get().get().toString()));
} else {
mono = client.post()
.body(BodyInserters.fromObject(body))
.retrieve().bodyToMono(String.class).map(response ->
{
// Care blocking operation! (use cacheManager -not found yet- prepared for reactive) cacheManager().getCache("hspad").putIfAbsent(body,response);
return ResponseEntity.ok(response);
});
}
return mono;
}
public static void main(String[] args) {
SpringApplication.run(GatewayApplication.class, args);
}
}
Spring CacheManagerで使用できるreactor cache add-on があります。しかし、受け入れられた回答のコメントで指摘されているように、現在、SpringキャッシュAPI(getsおよびputs)はまだブロックされています。この問題が解決されるまで、プログラムを完全にリアクティブにすることしかできません。
これがJavaのサンプルコードスニペットです。完全なサンプルプロジェクトは、githubの ここ です。
@Service
public class CatServiceImpl implements CatService {
private static final String CACHE_NAME = "sr";
private static final String KEY = "k";
@Autowired
private WebClient client;
@Autowired
private CacheManager cacheManager;
@SuppressWarnings("unchecked")
private Function<String, Mono<List<Signal<CatDto>>>> reader = k -> Mono
.justOrEmpty((Optional.ofNullable((List<CatDto>) (cacheManager.getCache(CACHE_NAME).get(k, List.class)))))
.flatMap(v -> Flux.fromIterable(v).materialize().collectList());
private BiFunction<String, List<Signal<CatDto>>, Mono<Void>> writer = (k, sigs) -> Flux.fromIterable(sigs)
.dematerialize().collectList().doOnNext(l -> cacheManager.getCache(CACHE_NAME).put(k, l)).then();
@Override
public Flux<CatDto> search() {
Flux<CatDto> fromServer = client.get().retrieve().bodyToFlux(CatDto.class);
return CacheFlux.lookup(reader, KEY).onCacheMissResume(fromServer).andWriteWith(writer);
}
}
今のところ、 @Cacheable
はFlux
(および一般的なReactor)では機能しません。しかし、あなたの例に関しては、メソッドを呼び出すたびに、新しいFlux
インスタンスを作成しているので、当然、何もキャッシュされません。
結果をキャッシュできるようにするには、Flux
をリストインスタンスに変換するか、単に1つのFlux
インスタンスを再利用し続ける必要があります。