Browse Source
This commit introduces JSON streaming support which consists of serializing HTTP request with application/stream+json media type as line delimited JSON. It also optimize Flux serialization for application/json by using flux.collectList() and a single Jackson invocation instead of one call per element previous strategy. This change result in a x4 throughput improvement for collection with a lot of small elements. Issues: SPR-15095, SPR-15104pull/1317/head
4 changed files with 194 additions and 33 deletions
@ -0,0 +1,148 @@ |
|||||||
|
/* |
||||||
|
* Copyright 2002-2017 the original author or authors. |
||||||
|
* |
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License"); |
||||||
|
* you may not use this file except in compliance with the License. |
||||||
|
* You may obtain a copy of the License at |
||||||
|
* |
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
* |
||||||
|
* Unless required by applicable law or agreed to in writing, software |
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS, |
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||||
|
* See the License for the specific language governing permissions and |
||||||
|
* limitations under the License. |
||||||
|
*/ |
||||||
|
|
||||||
|
package org.springframework.web.reactive.result.method.annotation; |
||||||
|
|
||||||
|
import java.time.Duration; |
||||||
|
|
||||||
|
import org.junit.Before; |
||||||
|
import org.junit.Test; |
||||||
|
import static org.springframework.http.MediaType.APPLICATION_STREAM_JSON; |
||||||
|
import static org.springframework.http.MediaType.APPLICATION_STREAM_JSON_VALUE; |
||||||
|
import reactor.core.publisher.Flux; |
||||||
|
import reactor.test.StepVerifier; |
||||||
|
|
||||||
|
import org.springframework.context.annotation.AnnotationConfigApplicationContext; |
||||||
|
import org.springframework.context.annotation.Bean; |
||||||
|
import org.springframework.context.annotation.Configuration; |
||||||
|
import org.springframework.http.server.reactive.AbstractHttpHandlerIntegrationTests; |
||||||
|
import org.springframework.http.server.reactive.HttpHandler; |
||||||
|
import org.springframework.web.bind.annotation.RequestMapping; |
||||||
|
import org.springframework.web.bind.annotation.RestController; |
||||||
|
import org.springframework.web.reactive.DispatcherHandler; |
||||||
|
import org.springframework.web.reactive.config.EnableWebFlux; |
||||||
|
import org.springframework.web.reactive.function.client.WebClient; |
||||||
|
import org.springframework.web.server.adapter.WebHttpHandlerBuilder; |
||||||
|
|
||||||
|
/** |
||||||
|
* @author Sebastien Deleuze |
||||||
|
*/ |
||||||
|
public class JsonStreamingIntegrationTests extends AbstractHttpHandlerIntegrationTests { |
||||||
|
|
||||||
|
private AnnotationConfigApplicationContext wac; |
||||||
|
|
||||||
|
private WebClient webClient; |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
@Before |
||||||
|
public void setup() throws Exception { |
||||||
|
super.setup(); |
||||||
|
this.webClient = WebClient.create("http://localhost:" + this.port); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
protected HttpHandler createHttpHandler() { |
||||||
|
this.wac = new AnnotationConfigApplicationContext(); |
||||||
|
this.wac.register(TestConfiguration.class); |
||||||
|
this.wac.refresh(); |
||||||
|
|
||||||
|
return WebHttpHandlerBuilder.webHandler(new DispatcherHandler(this.wac)).build(); |
||||||
|
} |
||||||
|
|
||||||
|
@Test |
||||||
|
public void jsonStreaming() throws Exception { |
||||||
|
Flux<Person> result = this.webClient.get() |
||||||
|
.uri("/stream") |
||||||
|
.accept(APPLICATION_STREAM_JSON) |
||||||
|
.exchange() |
||||||
|
.flatMap(response -> response.bodyToFlux(Person.class)); |
||||||
|
|
||||||
|
StepVerifier.create(result) |
||||||
|
.expectNext(new Person("foo 0")) |
||||||
|
.expectNext(new Person("foo 1")) |
||||||
|
.verifyComplete(); |
||||||
|
} |
||||||
|
|
||||||
|
@RestController |
||||||
|
@SuppressWarnings("unused") |
||||||
|
static class JsonStreamingController { |
||||||
|
|
||||||
|
@RequestMapping(value = "/stream", produces = APPLICATION_STREAM_JSON_VALUE) |
||||||
|
Flux<Person> person() { |
||||||
|
return Flux.interval(Duration.ofMillis(100)).map(l -> new Person("foo " + l)).take(2); |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
@Configuration |
||||||
|
@EnableWebFlux |
||||||
|
@SuppressWarnings("unused") |
||||||
|
static class TestConfiguration { |
||||||
|
|
||||||
|
@Bean |
||||||
|
public JsonStreamingController jsonStreamingController() { |
||||||
|
return new JsonStreamingController(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
private static class Person { |
||||||
|
|
||||||
|
private String name; |
||||||
|
|
||||||
|
@SuppressWarnings("unused") |
||||||
|
public Person() { |
||||||
|
} |
||||||
|
|
||||||
|
public Person(String name) { |
||||||
|
this.name = name; |
||||||
|
} |
||||||
|
|
||||||
|
public String getName() { |
||||||
|
return name; |
||||||
|
} |
||||||
|
|
||||||
|
public void setName(String name) { |
||||||
|
this.name = name; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public boolean equals(Object o) { |
||||||
|
if (this == o) { |
||||||
|
return true; |
||||||
|
} |
||||||
|
if (o == null || getClass() != o.getClass()) { |
||||||
|
return false; |
||||||
|
} |
||||||
|
Person person = (Person) o; |
||||||
|
return !(this.name != null ? !this.name.equals(person.name) : person.name != null); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public int hashCode() { |
||||||
|
return this.name != null ? this.name.hashCode() : 0; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public String toString() { |
||||||
|
return "Person{" + |
||||||
|
"name='" + name + '\'' + |
||||||
|
'}'; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
Loading…
Reference in new issue