Add Jackson Smile support to WebFlux
This binary format more efficient than JSON should be useful for server to server communication, for example in micro-services use cases. Issue: SPR-15424
This commit is contained in:
		
							parent
							
								
									50493a0f5f
								
							
						
					
					
						commit
						f46520e6e8
					
				| 
						 | 
					@ -952,6 +952,7 @@ project("spring-webflux") {
 | 
				
			||||||
		optional "javax.servlet:javax.servlet-api:${servletVersion}"
 | 
							optional "javax.servlet:javax.servlet-api:${servletVersion}"
 | 
				
			||||||
		optional("javax.xml.bind:jaxb-api:${jaxbVersion}")
 | 
							optional("javax.xml.bind:jaxb-api:${jaxbVersion}")
 | 
				
			||||||
		optional("com.fasterxml.jackson.core:jackson-databind:${jackson2Version}")
 | 
							optional("com.fasterxml.jackson.core:jackson-databind:${jackson2Version}")
 | 
				
			||||||
 | 
							optional("com.fasterxml.jackson.dataformat:jackson-dataformat-smile:${jackson2Version}")
 | 
				
			||||||
		optional("org.freemarker:freemarker:${freemarkerVersion}")
 | 
							optional("org.freemarker:freemarker:${freemarkerVersion}")
 | 
				
			||||||
		optional("org.apache.httpcomponents:httpclient:${httpclientVersion}") {
 | 
							optional("org.apache.httpcomponents:httpclient:${httpclientVersion}") {
 | 
				
			||||||
			exclude group: "commons-logging", module: "commons-logging"
 | 
								exclude group: "commons-logging", module: "commons-logging"
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -34,6 +34,8 @@ import org.springframework.core.codec.ResourceDecoder;
 | 
				
			||||||
import org.springframework.core.codec.StringDecoder;
 | 
					import org.springframework.core.codec.StringDecoder;
 | 
				
			||||||
import org.springframework.http.codec.json.Jackson2JsonDecoder;
 | 
					import org.springframework.http.codec.json.Jackson2JsonDecoder;
 | 
				
			||||||
import org.springframework.http.codec.json.Jackson2JsonEncoder;
 | 
					import org.springframework.http.codec.json.Jackson2JsonEncoder;
 | 
				
			||||||
 | 
					import org.springframework.http.codec.json.Jackson2SmileDecoder;
 | 
				
			||||||
 | 
					import org.springframework.http.codec.json.Jackson2SmileEncoder;
 | 
				
			||||||
import org.springframework.http.codec.xml.Jaxb2XmlDecoder;
 | 
					import org.springframework.http.codec.xml.Jaxb2XmlDecoder;
 | 
				
			||||||
import org.springframework.http.codec.xml.Jaxb2XmlEncoder;
 | 
					import org.springframework.http.codec.xml.Jaxb2XmlEncoder;
 | 
				
			||||||
import org.springframework.lang.Nullable;
 | 
					import org.springframework.lang.Nullable;
 | 
				
			||||||
| 
						 | 
					@ -54,6 +56,10 @@ abstract class AbstractCodecConfigurer implements CodecConfigurer {
 | 
				
			||||||
					ClassUtils.isPresent("com.fasterxml.jackson.core.JsonGenerator",
 | 
										ClassUtils.isPresent("com.fasterxml.jackson.core.JsonGenerator",
 | 
				
			||||||
							AbstractCodecConfigurer.class.getClassLoader());
 | 
												AbstractCodecConfigurer.class.getClassLoader());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						private static final boolean jackson2SmilePresent =
 | 
				
			||||||
 | 
								ClassUtils.isPresent("com.fasterxml.jackson.dataformat.smile.SmileFactory",
 | 
				
			||||||
 | 
										AbstractCodecConfigurer.class.getClassLoader());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	protected static final boolean jaxb2Present = ClassUtils.isPresent("javax.xml.bind.Binder",
 | 
						protected static final boolean jaxb2Present = ClassUtils.isPresent("javax.xml.bind.Binder",
 | 
				
			||||||
			AbstractCodecConfigurer.class.getClassLoader());
 | 
								AbstractCodecConfigurer.class.getClassLoader());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -119,10 +125,10 @@ abstract class AbstractCodecConfigurer implements CodecConfigurer {
 | 
				
			||||||
		private boolean registerDefaults = true;
 | 
							private boolean registerDefaults = true;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		@Nullable
 | 
							@Nullable
 | 
				
			||||||
		private Jackson2JsonDecoder jackson2Decoder;
 | 
							private Jackson2JsonDecoder jackson2JsonDecoder;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		@Nullable
 | 
							@Nullable
 | 
				
			||||||
		private Jackson2JsonEncoder jackson2Encoder;
 | 
							private Jackson2JsonEncoder jackson2JsonEncoder;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		@Nullable
 | 
							@Nullable
 | 
				
			||||||
		private DefaultCustomCodecs customCodecs;
 | 
							private DefaultCustomCodecs customCodecs;
 | 
				
			||||||
| 
						 | 
					@ -148,21 +154,21 @@ abstract class AbstractCodecConfigurer implements CodecConfigurer {
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		@Override
 | 
							@Override
 | 
				
			||||||
		public void jackson2Decoder(Jackson2JsonDecoder decoder) {
 | 
							public void jackson2JsonDecoder(Jackson2JsonDecoder decoder) {
 | 
				
			||||||
			this.jackson2Decoder = decoder;
 | 
								this.jackson2JsonDecoder = decoder;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		protected Jackson2JsonDecoder jackson2Decoder() {
 | 
							protected Jackson2JsonDecoder jackson2JsonDecoder() {
 | 
				
			||||||
			return (this.jackson2Decoder != null ? this.jackson2Decoder : new Jackson2JsonDecoder());
 | 
								return (this.jackson2JsonDecoder != null ? this.jackson2JsonDecoder : new Jackson2JsonDecoder());
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		@Override
 | 
							@Override
 | 
				
			||||||
		public void jackson2Encoder(Jackson2JsonEncoder encoder) {
 | 
							public void jackson2JsonEncoder(Jackson2JsonEncoder encoder) {
 | 
				
			||||||
			this.jackson2Encoder = encoder;
 | 
								this.jackson2JsonEncoder = encoder;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		protected Jackson2JsonEncoder jackson2Encoder() {
 | 
							protected Jackson2JsonEncoder jackson2JsonEncoder() {
 | 
				
			||||||
			return (this.jackson2Encoder != null ? this.jackson2Encoder : new Jackson2JsonEncoder());
 | 
								return (this.jackson2JsonEncoder != null ? this.jackson2JsonEncoder : new Jackson2JsonEncoder());
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		// Readers...
 | 
							// Readers...
 | 
				
			||||||
| 
						 | 
					@ -191,7 +197,10 @@ abstract class AbstractCodecConfigurer implements CodecConfigurer {
 | 
				
			||||||
				result.add(new DecoderHttpMessageReader<>(new Jaxb2XmlDecoder()));
 | 
									result.add(new DecoderHttpMessageReader<>(new Jaxb2XmlDecoder()));
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
			if (jackson2Present) {
 | 
								if (jackson2Present) {
 | 
				
			||||||
				result.add(new DecoderHttpMessageReader<>(jackson2Decoder()));
 | 
									result.add(new DecoderHttpMessageReader<>(jackson2JsonDecoder()));
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								if (jackson2SmilePresent) {
 | 
				
			||||||
 | 
									result.add(new DecoderHttpMessageReader<>(new Jackson2SmileDecoder()));
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
			return result;
 | 
								return result;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
| 
						 | 
					@ -229,7 +238,10 @@ abstract class AbstractCodecConfigurer implements CodecConfigurer {
 | 
				
			||||||
				result.add(new EncoderHttpMessageWriter<>(new Jaxb2XmlEncoder()));
 | 
									result.add(new EncoderHttpMessageWriter<>(new Jaxb2XmlEncoder()));
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
			if (jackson2Present) {
 | 
								if (jackson2Present) {
 | 
				
			||||||
				result.add(new EncoderHttpMessageWriter<>(jackson2Encoder()));
 | 
									result.add(new EncoderHttpMessageWriter<>(jackson2JsonEncoder()));
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								if (jackson2SmilePresent) {
 | 
				
			||||||
 | 
									result.add(new EncoderHttpMessageWriter<>(new Jackson2SmileEncoder()));
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
			return result;
 | 
								return result;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -64,7 +64,7 @@ public interface ClientCodecConfigurer extends CodecConfigurer {
 | 
				
			||||||
		/**
 | 
							/**
 | 
				
			||||||
		 * Configure the {@code Decoder} to use for Server-Sent Events.
 | 
							 * Configure the {@code Decoder} to use for Server-Sent Events.
 | 
				
			||||||
		 * <p>By default if this is not set, and Jackson is available, the
 | 
							 * <p>By default if this is not set, and Jackson is available, the
 | 
				
			||||||
		 * {@link #jackson2Decoder} override is used instead. Use this property
 | 
							 * {@link #jackson2JsonDecoder} override is used instead. Use this property
 | 
				
			||||||
		 * if you want to further customize the SSE decoder.
 | 
							 * if you want to further customize the SSE decoder.
 | 
				
			||||||
		 * @param decoder the decoder to use
 | 
							 * @param decoder the decoder to use
 | 
				
			||||||
		 */
 | 
							 */
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -74,13 +74,13 @@ public interface CodecConfigurer {
 | 
				
			||||||
		 * Override the default Jackson JSON {@code Decoder}.
 | 
							 * Override the default Jackson JSON {@code Decoder}.
 | 
				
			||||||
		 * @param decoder the decoder instance to use
 | 
							 * @param decoder the decoder instance to use
 | 
				
			||||||
		 */
 | 
							 */
 | 
				
			||||||
		void jackson2Decoder(Jackson2JsonDecoder decoder);
 | 
							void jackson2JsonDecoder(Jackson2JsonDecoder decoder);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		/**
 | 
							/**
 | 
				
			||||||
		 * Override the default Jackson JSON {@code Encoder}.
 | 
							 * Override the default Jackson JSON {@code Encoder}.
 | 
				
			||||||
		 * @param encoder the encoder instance to use
 | 
							 * @param encoder the encoder instance to use
 | 
				
			||||||
		 */
 | 
							 */
 | 
				
			||||||
		void jackson2Encoder(Jackson2JsonEncoder encoder);
 | 
							void jackson2JsonEncoder(Jackson2JsonEncoder encoder);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -83,7 +83,7 @@ class DefaultClientCodecConfigurer extends AbstractCodecConfigurer implements Cl
 | 
				
			||||||
			if (this.sseDecoder != null) {
 | 
								if (this.sseDecoder != null) {
 | 
				
			||||||
				return this.sseDecoder;
 | 
									return this.sseDecoder;
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
			return (jackson2Present ? jackson2Decoder() : null);
 | 
								return (jackson2Present ? jackson2JsonDecoder() : null);
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		@Override
 | 
							@Override
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -95,7 +95,7 @@ class DefaultServerCodecConfigurer extends AbstractCodecConfigurer implements Se
 | 
				
			||||||
			if (this.sseEncoder != null) {
 | 
								if (this.sseEncoder != null) {
 | 
				
			||||||
				return this.sseEncoder;
 | 
									return this.sseEncoder;
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
			return jackson2Present ? jackson2Encoder() : null;
 | 
								return jackson2Present ? jackson2JsonEncoder() : null;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -55,7 +55,7 @@ public interface ServerCodecConfigurer extends CodecConfigurer {
 | 
				
			||||||
		/**
 | 
							/**
 | 
				
			||||||
		 * Configure the {@code Encoder} to use for Server-Sent Events.
 | 
							 * Configure the {@code Encoder} to use for Server-Sent Events.
 | 
				
			||||||
		 * <p>By default if this is not set, and Jackson is available, the
 | 
							 * <p>By default if this is not set, and Jackson is available, the
 | 
				
			||||||
		 * {@link #jackson2Encoder} override is used instead. Use this property
 | 
							 * {@link #jackson2JsonEncoder} override is used instead. Use this property
 | 
				
			||||||
		 * if you want to further customize the SSE encoder.
 | 
							 * if you want to further customize the SSE encoder.
 | 
				
			||||||
		 */
 | 
							 */
 | 
				
			||||||
		void serverSentEventEncoder(Encoder<?> encoder);
 | 
							void serverSentEventEncoder(Encoder<?> encoder);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,149 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * 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.http.codec.json;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import java.io.IOException;
 | 
				
			||||||
 | 
					import java.io.UncheckedIOException;
 | 
				
			||||||
 | 
					import java.lang.annotation.Annotation;
 | 
				
			||||||
 | 
					import java.util.Map;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.core.JsonFactory;
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.core.JsonParser;
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.core.JsonProcessingException;
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.databind.JavaType;
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.databind.ObjectMapper;
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.databind.ObjectReader;
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.databind.exc.InvalidDefinitionException;
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.databind.util.TokenBuffer;
 | 
				
			||||||
 | 
					import org.reactivestreams.Publisher;
 | 
				
			||||||
 | 
					import reactor.core.publisher.Flux;
 | 
				
			||||||
 | 
					import reactor.core.publisher.Mono;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import org.springframework.core.MethodParameter;
 | 
				
			||||||
 | 
					import org.springframework.core.ResolvableType;
 | 
				
			||||||
 | 
					import org.springframework.core.codec.CodecException;
 | 
				
			||||||
 | 
					import org.springframework.core.codec.DecodingException;
 | 
				
			||||||
 | 
					import org.springframework.core.io.buffer.DataBuffer;
 | 
				
			||||||
 | 
					import org.springframework.http.codec.HttpMessageDecoder;
 | 
				
			||||||
 | 
					import org.springframework.http.server.reactive.ServerHttpRequest;
 | 
				
			||||||
 | 
					import org.springframework.http.server.reactive.ServerHttpResponse;
 | 
				
			||||||
 | 
					import org.springframework.lang.Nullable;
 | 
				
			||||||
 | 
					import org.springframework.util.Assert;
 | 
				
			||||||
 | 
					import org.springframework.util.MimeType;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * Base class providing support methods for Jackson 2.9 decoding.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * @author Sebastien Deleuze
 | 
				
			||||||
 | 
					 * @author Rossen Stoyanchev
 | 
				
			||||||
 | 
					 * @since 5.0
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					public abstract class AbstractJackson2Decoder extends Jackson2CodecSupport implements HttpMessageDecoder<Object> {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/**
 | 
				
			||||||
 | 
						 * Constructor with a Jackson {@link ObjectMapper} to use.
 | 
				
			||||||
 | 
						 */
 | 
				
			||||||
 | 
						protected AbstractJackson2Decoder(ObjectMapper mapper, MimeType... mimeTypes) {
 | 
				
			||||||
 | 
							super(mapper, mimeTypes);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Override
 | 
				
			||||||
 | 
						public boolean canDecode(ResolvableType elementType, @Nullable MimeType mimeType) {
 | 
				
			||||||
 | 
							JavaType javaType = objectMapper().getTypeFactory().constructType(elementType.getType());
 | 
				
			||||||
 | 
							// Skip String: CharSequenceDecoder + "*/*" comes after
 | 
				
			||||||
 | 
							return (!CharSequence.class.isAssignableFrom(elementType.resolve(Object.class)) &&
 | 
				
			||||||
 | 
									objectMapper().canDeserialize(javaType) && supportsMimeType(mimeType));
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Override
 | 
				
			||||||
 | 
						public Flux<Object> decode(Publisher<DataBuffer> input, ResolvableType elementType,
 | 
				
			||||||
 | 
								@Nullable MimeType mimeType, @Nullable Map<String, Object> hints) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							Flux<TokenBuffer> tokens = tokenize(input, true);
 | 
				
			||||||
 | 
							return decodeInternal(tokens, elementType, mimeType, hints);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Override
 | 
				
			||||||
 | 
						public Mono<Object> decodeToMono(Publisher<DataBuffer> input, ResolvableType elementType,
 | 
				
			||||||
 | 
								@Nullable MimeType mimeType, @Nullable Map<String, Object> hints) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							Flux<TokenBuffer> tokens = tokenize(input, false);
 | 
				
			||||||
 | 
							return decodeInternal(tokens, elementType, mimeType, hints).singleOrEmpty();
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						private Flux<TokenBuffer> tokenize(Publisher<DataBuffer> input, boolean tokenizeArrayElements) {
 | 
				
			||||||
 | 
							try {
 | 
				
			||||||
 | 
								JsonFactory factory = objectMapper().getFactory();
 | 
				
			||||||
 | 
								JsonParser nonBlockingParser = factory.createNonBlockingByteArrayParser();
 | 
				
			||||||
 | 
								Jackson2Tokenizer tokenizer = new Jackson2Tokenizer(nonBlockingParser,
 | 
				
			||||||
 | 
										tokenizeArrayElements);
 | 
				
			||||||
 | 
								return Flux.from(input)
 | 
				
			||||||
 | 
										.flatMap(tokenizer)
 | 
				
			||||||
 | 
										.doFinally(t -> tokenizer.endOfInput());
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							catch (IOException ex) {
 | 
				
			||||||
 | 
								return Flux.error(new UncheckedIOException(ex));
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						private Flux<Object> decodeInternal(Flux<TokenBuffer> tokens,
 | 
				
			||||||
 | 
								ResolvableType elementType, @Nullable MimeType mimeType,
 | 
				
			||||||
 | 
								@Nullable Map<String, Object> hints) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							Assert.notNull(tokens, "'tokens' must not be null");
 | 
				
			||||||
 | 
							Assert.notNull(elementType, "'elementType' must not be null");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							MethodParameter param = getParameter(elementType);
 | 
				
			||||||
 | 
							Class<?> contextClass = (param != null ? param.getContainingClass() : null);
 | 
				
			||||||
 | 
							JavaType javaType = getJavaType(elementType.getType(), contextClass);
 | 
				
			||||||
 | 
							Class<?> jsonView = (hints != null ? (Class<?>) hints.get(Jackson2CodecSupport.JSON_VIEW_HINT) : null);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							ObjectReader reader = (jsonView != null ?
 | 
				
			||||||
 | 
									objectMapper().readerWithView(jsonView).forType(javaType) :
 | 
				
			||||||
 | 
									objectMapper().readerFor(javaType));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							return tokens.map(tokenBuffer -> {
 | 
				
			||||||
 | 
								try {
 | 
				
			||||||
 | 
									return reader.readValue(tokenBuffer.asParser());
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								catch (InvalidDefinitionException ex) {
 | 
				
			||||||
 | 
									throw new CodecException("Type definition error: " + ex.getType(), ex);
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								catch (JsonProcessingException ex) {
 | 
				
			||||||
 | 
									throw new DecodingException("JSON decoding error: " + ex.getOriginalMessage(), ex);
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								catch (IOException ex) {
 | 
				
			||||||
 | 
									throw new DecodingException("I/O error while parsing input stream", ex);
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							});
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// HttpMessageDecoder...
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Override
 | 
				
			||||||
 | 
						public Map<String, Object> getDecodeHints(ResolvableType actualType, ResolvableType elementType,
 | 
				
			||||||
 | 
								ServerHttpRequest request, ServerHttpResponse response) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							return getHints(actualType);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Override
 | 
				
			||||||
 | 
						protected <A extends Annotation> A getAnnotation(MethodParameter parameter, Class<A> annotType) {
 | 
				
			||||||
 | 
							return parameter.getParameterAnnotation(annotType);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,170 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * 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.http.codec.json;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import java.io.IOException;
 | 
				
			||||||
 | 
					import java.io.OutputStream;
 | 
				
			||||||
 | 
					import java.lang.annotation.Annotation;
 | 
				
			||||||
 | 
					import java.util.ArrayList;
 | 
				
			||||||
 | 
					import java.util.Collections;
 | 
				
			||||||
 | 
					import java.util.List;
 | 
				
			||||||
 | 
					import java.util.Map;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.core.JsonProcessingException;
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.databind.JavaType;
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.databind.ObjectMapper;
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.databind.ObjectWriter;
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.databind.exc.InvalidDefinitionException;
 | 
				
			||||||
 | 
					import org.reactivestreams.Publisher;
 | 
				
			||||||
 | 
					import reactor.core.publisher.Flux;
 | 
				
			||||||
 | 
					import reactor.core.publisher.Mono;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import org.springframework.core.MethodParameter;
 | 
				
			||||||
 | 
					import org.springframework.core.ResolvableType;
 | 
				
			||||||
 | 
					import org.springframework.core.codec.CodecException;
 | 
				
			||||||
 | 
					import org.springframework.core.codec.EncodingException;
 | 
				
			||||||
 | 
					import org.springframework.core.io.buffer.DataBuffer;
 | 
				
			||||||
 | 
					import org.springframework.core.io.buffer.DataBufferFactory;
 | 
				
			||||||
 | 
					import org.springframework.http.MediaType;
 | 
				
			||||||
 | 
					import org.springframework.http.codec.HttpMessageEncoder;
 | 
				
			||||||
 | 
					import org.springframework.http.server.reactive.ServerHttpRequest;
 | 
				
			||||||
 | 
					import org.springframework.http.server.reactive.ServerHttpResponse;
 | 
				
			||||||
 | 
					import org.springframework.lang.Nullable;
 | 
				
			||||||
 | 
					import org.springframework.util.Assert;
 | 
				
			||||||
 | 
					import org.springframework.util.MimeType;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * Base class providing support methods for Jackson 2.9 encoding.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * @author Sebastien Deleuze
 | 
				
			||||||
 | 
					 * @author Arjen Poutsma
 | 
				
			||||||
 | 
					 * @since 5.0
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					public abstract class AbstractJackson2Encoder extends Jackson2CodecSupport implements HttpMessageEncoder<Object> {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						protected final List<MediaType> streamingMediaTypes = new ArrayList<>(1);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/**
 | 
				
			||||||
 | 
						 * Constructor with a Jackson {@link ObjectMapper} to use.
 | 
				
			||||||
 | 
						 */
 | 
				
			||||||
 | 
						protected AbstractJackson2Encoder(ObjectMapper mapper, MimeType... mimeTypes) {
 | 
				
			||||||
 | 
							super(mapper, mimeTypes);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/**
 | 
				
			||||||
 | 
						 * Configure "streaming" media types for which flushing should be performed
 | 
				
			||||||
 | 
						 * automatically vs at the end of the stream.
 | 
				
			||||||
 | 
						 * <p>By default this is set to {@link MediaType#APPLICATION_STREAM_JSON}.
 | 
				
			||||||
 | 
						 * @param mediaTypes one or more media types to add to the list
 | 
				
			||||||
 | 
						 * @see HttpMessageEncoder#getStreamingMediaTypes()
 | 
				
			||||||
 | 
						 */
 | 
				
			||||||
 | 
						public void setStreamingMediaTypes(List<MediaType> mediaTypes) {
 | 
				
			||||||
 | 
							this.streamingMediaTypes.clear();
 | 
				
			||||||
 | 
							this.streamingMediaTypes.addAll(mediaTypes);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Override
 | 
				
			||||||
 | 
						public boolean canEncode(ResolvableType elementType, @Nullable MimeType mimeType) {
 | 
				
			||||||
 | 
							Class<?> clazz = elementType.resolve(Object.class);
 | 
				
			||||||
 | 
							return (Object.class == clazz) ||
 | 
				
			||||||
 | 
									!String.class.isAssignableFrom(elementType.resolve(clazz)) &&
 | 
				
			||||||
 | 
											objectMapper().canSerialize(clazz) && supportsMimeType(mimeType);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Override
 | 
				
			||||||
 | 
						public Flux<DataBuffer> encode(Publisher<?> inputStream, DataBufferFactory bufferFactory,
 | 
				
			||||||
 | 
								ResolvableType elementType, @Nullable MimeType mimeType, @Nullable Map<String, Object> hints) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							Assert.notNull(inputStream, "'inputStream' must not be null");
 | 
				
			||||||
 | 
							Assert.notNull(bufferFactory, "'bufferFactory' must not be null");
 | 
				
			||||||
 | 
							Assert.notNull(elementType, "'elementType' must not be null");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (inputStream instanceof Mono) {
 | 
				
			||||||
 | 
								return Flux.from(inputStream).map(value ->
 | 
				
			||||||
 | 
										encodeValue(value, mimeType, bufferFactory, elementType, hints));
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							else if (this.streamingMediaTypes.stream().anyMatch(streamingMediaType -> streamingMediaType.isCompatibleWith(mimeType))) {
 | 
				
			||||||
 | 
								return Flux.from(inputStream).map(value -> {
 | 
				
			||||||
 | 
									DataBuffer buffer = encodeValue(value, mimeType, bufferFactory, elementType, hints);
 | 
				
			||||||
 | 
									buffer.write(new byte[]{'\n'});
 | 
				
			||||||
 | 
									return buffer;
 | 
				
			||||||
 | 
								});
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							else {
 | 
				
			||||||
 | 
								ResolvableType listType = ResolvableType.forClassWithGenerics(List.class, elementType);
 | 
				
			||||||
 | 
								return Flux.from(inputStream).collectList().map(list ->
 | 
				
			||||||
 | 
										encodeValue(list, mimeType, bufferFactory, listType, hints)).flux();
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						private DataBuffer encodeValue(Object value, @Nullable MimeType mimeType, DataBufferFactory bufferFactory,
 | 
				
			||||||
 | 
								ResolvableType elementType, @Nullable Map<String, Object> hints) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							JavaType javaType = getJavaType(elementType.getType(), null);
 | 
				
			||||||
 | 
							Class<?> jsonView = (hints != null ? (Class<?>) hints.get(Jackson2CodecSupport.JSON_VIEW_HINT) : null);
 | 
				
			||||||
 | 
							ObjectWriter writer = (jsonView != null ?
 | 
				
			||||||
 | 
									objectMapper().writerWithView(jsonView) : objectMapper().writer());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (javaType.isContainerType()) {
 | 
				
			||||||
 | 
								writer = writer.forType(javaType);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							writer = customizeWriter(writer, mimeType, elementType, hints);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							DataBuffer buffer = bufferFactory.allocateBuffer();
 | 
				
			||||||
 | 
							OutputStream outputStream = buffer.asOutputStream();
 | 
				
			||||||
 | 
							try {
 | 
				
			||||||
 | 
								writer.writeValue(outputStream, value);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							catch (InvalidDefinitionException ex) {
 | 
				
			||||||
 | 
								throw new CodecException("Type definition error: " + ex.getType(), ex);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							catch (JsonProcessingException ex) {
 | 
				
			||||||
 | 
								throw new EncodingException("JSON encoding error: " + ex.getOriginalMessage(), ex);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							catch (IOException ex) {
 | 
				
			||||||
 | 
								throw new IllegalStateException("Unexpected I/O error while writing to data buffer", ex);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							return buffer;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						
 | 
				
			||||||
 | 
						protected ObjectWriter customizeWriter(ObjectWriter writer, @Nullable MimeType mimeType,
 | 
				
			||||||
 | 
								ResolvableType elementType, @Nullable Map<String, Object> hints) {
 | 
				
			||||||
 | 
							return writer;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// HttpMessageEncoder...
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Override
 | 
				
			||||||
 | 
						public List<MediaType> getStreamingMediaTypes() {
 | 
				
			||||||
 | 
							return Collections.unmodifiableList(this.streamingMediaTypes);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Override
 | 
				
			||||||
 | 
						public Map<String, Object> getEncodeHints(@Nullable ResolvableType actualType, ResolvableType elementType,
 | 
				
			||||||
 | 
								@Nullable MediaType mediaType, ServerHttpRequest request, ServerHttpResponse response) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							return (actualType != null ? getHints(actualType) : Collections.emptyMap());
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Override
 | 
				
			||||||
 | 
						protected <A extends Annotation> A getAnnotation(MethodParameter parameter, Class<A> annotType) {
 | 
				
			||||||
 | 
							return parameter.getMethodAnnotation(annotType);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -16,35 +16,10 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
package org.springframework.http.codec.json;
 | 
					package org.springframework.http.codec.json;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import java.io.IOException;
 | 
					 | 
				
			||||||
import java.io.UncheckedIOException;
 | 
					 | 
				
			||||||
import java.lang.annotation.Annotation;
 | 
					 | 
				
			||||||
import java.util.List;
 | 
					import java.util.List;
 | 
				
			||||||
import java.util.Map;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
import com.fasterxml.jackson.core.JsonFactory;
 | 
					 | 
				
			||||||
import com.fasterxml.jackson.core.JsonParser;
 | 
					 | 
				
			||||||
import com.fasterxml.jackson.core.JsonProcessingException;
 | 
					 | 
				
			||||||
import com.fasterxml.jackson.databind.JavaType;
 | 
					 | 
				
			||||||
import com.fasterxml.jackson.databind.ObjectMapper;
 | 
					import com.fasterxml.jackson.databind.ObjectMapper;
 | 
				
			||||||
import com.fasterxml.jackson.databind.ObjectReader;
 | 
					 | 
				
			||||||
import com.fasterxml.jackson.databind.exc.InvalidDefinitionException;
 | 
					 | 
				
			||||||
import com.fasterxml.jackson.databind.util.TokenBuffer;
 | 
					 | 
				
			||||||
import org.reactivestreams.Publisher;
 | 
					 | 
				
			||||||
import reactor.core.publisher.Flux;
 | 
					 | 
				
			||||||
import reactor.core.publisher.Mono;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import org.springframework.core.MethodParameter;
 | 
					 | 
				
			||||||
import org.springframework.core.ResolvableType;
 | 
					 | 
				
			||||||
import org.springframework.core.codec.CodecException;
 | 
					 | 
				
			||||||
import org.springframework.core.codec.DecodingException;
 | 
					 | 
				
			||||||
import org.springframework.core.io.buffer.DataBuffer;
 | 
					 | 
				
			||||||
import org.springframework.http.codec.HttpMessageDecoder;
 | 
					 | 
				
			||||||
import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
 | 
					import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
 | 
				
			||||||
import org.springframework.http.server.reactive.ServerHttpRequest;
 | 
					 | 
				
			||||||
import org.springframework.http.server.reactive.ServerHttpResponse;
 | 
					 | 
				
			||||||
import org.springframework.lang.Nullable;
 | 
					 | 
				
			||||||
import org.springframework.util.Assert;
 | 
					 | 
				
			||||||
import org.springframework.util.MimeType;
 | 
					import org.springframework.util.MimeType;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/**
 | 
					/**
 | 
				
			||||||
| 
						 | 
					@ -55,7 +30,7 @@ import org.springframework.util.MimeType;
 | 
				
			||||||
 * @since 5.0
 | 
					 * @since 5.0
 | 
				
			||||||
 * @see Jackson2JsonEncoder
 | 
					 * @see Jackson2JsonEncoder
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
public class Jackson2JsonDecoder extends Jackson2CodecSupport implements HttpMessageDecoder<Object> {
 | 
					public class Jackson2JsonDecoder extends AbstractJackson2Decoder {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	public Jackson2JsonDecoder() {
 | 
						public Jackson2JsonDecoder() {
 | 
				
			||||||
		super(Jackson2ObjectMapperBuilder.json().build());
 | 
							super(Jackson2ObjectMapperBuilder.json().build());
 | 
				
			||||||
| 
						 | 
					@ -65,96 +40,8 @@ public class Jackson2JsonDecoder extends Jackson2CodecSupport implements HttpMes
 | 
				
			||||||
		super(mapper, mimeTypes);
 | 
							super(mapper, mimeTypes);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	@Override
 | 
					 | 
				
			||||||
	public boolean canDecode(ResolvableType elementType, @Nullable MimeType mimeType) {
 | 
					 | 
				
			||||||
		JavaType javaType = objectMapper().getTypeFactory().constructType(elementType.getType());
 | 
					 | 
				
			||||||
		// Skip String: CharSequenceDecoder + "*/*" comes after
 | 
					 | 
				
			||||||
		return (!CharSequence.class.isAssignableFrom(elementType.resolve(Object.class)) &&
 | 
					 | 
				
			||||||
				objectMapper().canDeserialize(javaType) && supportsMimeType(mimeType));
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	@Override
 | 
						@Override
 | 
				
			||||||
	public List<MimeType> getDecodableMimeTypes() {
 | 
						public List<MimeType> getDecodableMimeTypes() {
 | 
				
			||||||
		return JSON_MIME_TYPES;
 | 
							return JSON_MIME_TYPES;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					 | 
				
			||||||
	@Override
 | 
					 | 
				
			||||||
	public Flux<Object> decode(Publisher<DataBuffer> input, ResolvableType elementType,
 | 
					 | 
				
			||||||
			@Nullable MimeType mimeType, @Nullable Map<String, Object> hints) {
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		Flux<TokenBuffer> tokens = tokenize(input, true);
 | 
					 | 
				
			||||||
		return decodeInternal(tokens, elementType, mimeType, hints);
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	@Override
 | 
					 | 
				
			||||||
	public Mono<Object> decodeToMono(Publisher<DataBuffer> input, ResolvableType elementType,
 | 
					 | 
				
			||||||
			@Nullable MimeType mimeType, @Nullable Map<String, Object> hints) {
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		Flux<TokenBuffer> tokens = tokenize(input, false);
 | 
					 | 
				
			||||||
		return decodeInternal(tokens, elementType, mimeType, hints).singleOrEmpty();
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	private Flux<TokenBuffer> tokenize(Publisher<DataBuffer> input, boolean tokenizeArrayElements) {
 | 
					 | 
				
			||||||
		try {
 | 
					 | 
				
			||||||
			JsonFactory factory = objectMapper().getFactory();
 | 
					 | 
				
			||||||
			JsonParser nonBlockingParser = factory.createNonBlockingByteArrayParser();
 | 
					 | 
				
			||||||
			Jackson2Tokenizer tokenizer = new Jackson2Tokenizer(nonBlockingParser,
 | 
					 | 
				
			||||||
					tokenizeArrayElements);
 | 
					 | 
				
			||||||
			return Flux.from(input)
 | 
					 | 
				
			||||||
					.flatMap(tokenizer)
 | 
					 | 
				
			||||||
					.doFinally(t -> tokenizer.endOfInput());
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
		catch (IOException ex) {
 | 
					 | 
				
			||||||
			return Flux.error(new UncheckedIOException(ex));
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	private Flux<Object> decodeInternal(Flux<TokenBuffer> tokens,
 | 
					 | 
				
			||||||
			ResolvableType elementType, @Nullable MimeType mimeType,
 | 
					 | 
				
			||||||
			@Nullable Map<String, Object> hints) {
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		Assert.notNull(tokens, "'tokens' must not be null");
 | 
					 | 
				
			||||||
		Assert.notNull(elementType, "'elementType' must not be null");
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		MethodParameter param = getParameter(elementType);
 | 
					 | 
				
			||||||
		Class<?> contextClass = (param != null ? param.getContainingClass() : null);
 | 
					 | 
				
			||||||
		JavaType javaType = getJavaType(elementType.getType(), contextClass);
 | 
					 | 
				
			||||||
		Class<?> jsonView = (hints != null ? (Class<?>) hints.get(Jackson2CodecSupport.JSON_VIEW_HINT) : null);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		ObjectReader reader = (jsonView != null ?
 | 
					 | 
				
			||||||
				objectMapper().readerWithView(jsonView).forType(javaType) :
 | 
					 | 
				
			||||||
				objectMapper().readerFor(javaType));
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		return tokens.map(tokenBuffer -> {
 | 
					 | 
				
			||||||
			try {
 | 
					 | 
				
			||||||
				return reader.readValue(tokenBuffer.asParser());
 | 
					 | 
				
			||||||
			}
 | 
					 | 
				
			||||||
			catch (InvalidDefinitionException ex) {
 | 
					 | 
				
			||||||
				throw new CodecException("Type definition error: " + ex.getType(), ex);
 | 
					 | 
				
			||||||
			}
 | 
					 | 
				
			||||||
			catch (JsonProcessingException ex) {
 | 
					 | 
				
			||||||
				throw new DecodingException("JSON decoding error: " + ex.getOriginalMessage(), ex);
 | 
					 | 
				
			||||||
			}
 | 
					 | 
				
			||||||
			catch (IOException ex) {
 | 
					 | 
				
			||||||
				throw new DecodingException("I/O error while parsing input stream", ex);
 | 
					 | 
				
			||||||
			}
 | 
					 | 
				
			||||||
		});
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	// HttpMessageDecoder...
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	@Override
 | 
					 | 
				
			||||||
	public Map<String, Object> getDecodeHints(ResolvableType actualType, ResolvableType elementType,
 | 
					 | 
				
			||||||
			ServerHttpRequest request, ServerHttpResponse response) {
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		return getHints(actualType);
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	@Override
 | 
					 | 
				
			||||||
	protected <A extends Annotation> A getAnnotation(MethodParameter parameter, Class<A> annotType) {
 | 
					 | 
				
			||||||
		return parameter.getParameterAnnotation(annotType);
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -16,40 +16,20 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
package org.springframework.http.codec.json;
 | 
					package org.springframework.http.codec.json;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import java.io.IOException;
 | 
					 | 
				
			||||||
import java.io.OutputStream;
 | 
					 | 
				
			||||||
import java.lang.annotation.Annotation;
 | 
					 | 
				
			||||||
import java.util.ArrayList;
 | 
					 | 
				
			||||||
import java.util.Collections;
 | 
					 | 
				
			||||||
import java.util.List;
 | 
					import java.util.List;
 | 
				
			||||||
import java.util.Map;
 | 
					import java.util.Map;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import com.fasterxml.jackson.core.JsonProcessingException;
 | 
					 | 
				
			||||||
import com.fasterxml.jackson.core.PrettyPrinter;
 | 
					import com.fasterxml.jackson.core.PrettyPrinter;
 | 
				
			||||||
import com.fasterxml.jackson.core.util.DefaultIndenter;
 | 
					import com.fasterxml.jackson.core.util.DefaultIndenter;
 | 
				
			||||||
import com.fasterxml.jackson.core.util.DefaultPrettyPrinter;
 | 
					import com.fasterxml.jackson.core.util.DefaultPrettyPrinter;
 | 
				
			||||||
import com.fasterxml.jackson.databind.JavaType;
 | 
					 | 
				
			||||||
import com.fasterxml.jackson.databind.ObjectMapper;
 | 
					import com.fasterxml.jackson.databind.ObjectMapper;
 | 
				
			||||||
import com.fasterxml.jackson.databind.ObjectWriter;
 | 
					import com.fasterxml.jackson.databind.ObjectWriter;
 | 
				
			||||||
import com.fasterxml.jackson.databind.SerializationFeature;
 | 
					import com.fasterxml.jackson.databind.SerializationFeature;
 | 
				
			||||||
import com.fasterxml.jackson.databind.exc.InvalidDefinitionException;
 | 
					 | 
				
			||||||
import org.reactivestreams.Publisher;
 | 
					 | 
				
			||||||
import reactor.core.publisher.Flux;
 | 
					 | 
				
			||||||
import reactor.core.publisher.Mono;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
import org.springframework.core.MethodParameter;
 | 
					 | 
				
			||||||
import org.springframework.core.ResolvableType;
 | 
					import org.springframework.core.ResolvableType;
 | 
				
			||||||
import org.springframework.core.codec.CodecException;
 | 
					 | 
				
			||||||
import org.springframework.core.codec.EncodingException;
 | 
					 | 
				
			||||||
import org.springframework.core.io.buffer.DataBuffer;
 | 
					 | 
				
			||||||
import org.springframework.core.io.buffer.DataBufferFactory;
 | 
					 | 
				
			||||||
import org.springframework.http.MediaType;
 | 
					import org.springframework.http.MediaType;
 | 
				
			||||||
import org.springframework.http.codec.HttpMessageEncoder;
 | 
					 | 
				
			||||||
import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
 | 
					import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
 | 
				
			||||||
import org.springframework.http.server.reactive.ServerHttpRequest;
 | 
					 | 
				
			||||||
import org.springframework.http.server.reactive.ServerHttpResponse;
 | 
					 | 
				
			||||||
import org.springframework.lang.Nullable;
 | 
					import org.springframework.lang.Nullable;
 | 
				
			||||||
import org.springframework.util.Assert;
 | 
					 | 
				
			||||||
import org.springframework.util.MimeType;
 | 
					import org.springframework.util.MimeType;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/**
 | 
					/**
 | 
				
			||||||
| 
						 | 
					@ -61,13 +41,10 @@ import org.springframework.util.MimeType;
 | 
				
			||||||
 * @since 5.0
 | 
					 * @since 5.0
 | 
				
			||||||
 * @see Jackson2JsonDecoder
 | 
					 * @see Jackson2JsonDecoder
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
public class Jackson2JsonEncoder extends Jackson2CodecSupport implements HttpMessageEncoder<Object> {
 | 
					public class Jackson2JsonEncoder extends AbstractJackson2Encoder {
 | 
				
			||||||
 | 
						
 | 
				
			||||||
	private final List<MediaType> streamingMediaTypes = new ArrayList<>(1);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	private final PrettyPrinter ssePrettyPrinter;
 | 
						private final PrettyPrinter ssePrettyPrinter;
 | 
				
			||||||
 | 
						
 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	public Jackson2JsonEncoder() {
 | 
						public Jackson2JsonEncoder() {
 | 
				
			||||||
		this(Jackson2ObjectMapperBuilder.json().build());
 | 
							this(Jackson2ObjectMapperBuilder.json().build());
 | 
				
			||||||
| 
						 | 
					@ -85,113 +62,16 @@ public class Jackson2JsonEncoder extends Jackson2CodecSupport implements HttpMes
 | 
				
			||||||
		return printer;
 | 
							return printer;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Override
 | 
				
			||||||
	/**
 | 
						protected ObjectWriter customizeWriter(ObjectWriter writer, @Nullable MimeType mimeType,
 | 
				
			||||||
	 * Configure "streaming" media types for which flushing should be performed
 | 
								ResolvableType elementType, @Nullable Map<String, Object> hints) {
 | 
				
			||||||
	 * automatically vs at the end of the stream.
 | 
							
 | 
				
			||||||
	 * <p>By default this is set to {@link MediaType#APPLICATION_STREAM_JSON}.
 | 
							return (this.ssePrettyPrinter != null && MediaType.TEXT_EVENT_STREAM.isCompatibleWith(mimeType) &&
 | 
				
			||||||
	 * @param mediaTypes one or more media types to add to the list
 | 
									writer.getConfig().isEnabled(SerializationFeature.INDENT_OUTPUT) ? writer.with(this.ssePrettyPrinter) : writer);
 | 
				
			||||||
	 * @see HttpMessageEncoder#getStreamingMediaTypes()
 | 
					 | 
				
			||||||
	 */
 | 
					 | 
				
			||||||
	public void setStreamingMediaTypes(List<MediaType> mediaTypes) {
 | 
					 | 
				
			||||||
		this.streamingMediaTypes.clear();
 | 
					 | 
				
			||||||
		this.streamingMediaTypes.addAll(mediaTypes);
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	@Override
 | 
						@Override
 | 
				
			||||||
	public List<MimeType> getEncodableMimeTypes() {
 | 
						public List<MimeType> getEncodableMimeTypes() {
 | 
				
			||||||
		return JSON_MIME_TYPES;
 | 
							return JSON_MIME_TYPES;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	@Override
 | 
					 | 
				
			||||||
	public boolean canEncode(ResolvableType elementType, @Nullable MimeType mimeType) {
 | 
					 | 
				
			||||||
		Class<?> clazz = elementType.resolve(Object.class);
 | 
					 | 
				
			||||||
		return (Object.class == clazz) ||
 | 
					 | 
				
			||||||
				!String.class.isAssignableFrom(elementType.resolve(clazz)) &&
 | 
					 | 
				
			||||||
				objectMapper().canSerialize(clazz) && supportsMimeType(mimeType);
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	@Override
 | 
					 | 
				
			||||||
	public Flux<DataBuffer> encode(Publisher<?> inputStream, DataBufferFactory bufferFactory,
 | 
					 | 
				
			||||||
			ResolvableType elementType, @Nullable MimeType mimeType, @Nullable Map<String, Object> hints) {
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		Assert.notNull(inputStream, "'inputStream' must not be null");
 | 
					 | 
				
			||||||
		Assert.notNull(bufferFactory, "'bufferFactory' must not be null");
 | 
					 | 
				
			||||||
		Assert.notNull(elementType, "'elementType' must not be null");
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		if (inputStream instanceof Mono) {
 | 
					 | 
				
			||||||
			return Flux.from(inputStream).map(value ->
 | 
					 | 
				
			||||||
					encodeValue(value, mimeType, bufferFactory, elementType, hints));
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
		else if (this.streamingMediaTypes.stream().anyMatch(streamingMediaType -> streamingMediaType.isCompatibleWith(mimeType))) {
 | 
					 | 
				
			||||||
			return Flux.from(inputStream).map(value -> {
 | 
					 | 
				
			||||||
				DataBuffer buffer = encodeValue(value, mimeType, bufferFactory, elementType, hints);
 | 
					 | 
				
			||||||
				buffer.write(new byte[]{'\n'});
 | 
					 | 
				
			||||||
				return buffer;
 | 
					 | 
				
			||||||
			});
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
		else {
 | 
					 | 
				
			||||||
			ResolvableType listType = ResolvableType.forClassWithGenerics(List.class, elementType);
 | 
					 | 
				
			||||||
			return Flux.from(inputStream).collectList().map(list ->
 | 
					 | 
				
			||||||
					encodeValue(list, mimeType, bufferFactory, listType, hints)).flux();
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	private DataBuffer encodeValue(Object value, @Nullable MimeType mimeType, DataBufferFactory bufferFactory,
 | 
					 | 
				
			||||||
			ResolvableType elementType, @Nullable Map<String, Object> hints) {
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		JavaType javaType = getJavaType(elementType.getType(), null);
 | 
					 | 
				
			||||||
		Class<?> jsonView = (hints != null ? (Class<?>) hints.get(Jackson2CodecSupport.JSON_VIEW_HINT) : null);
 | 
					 | 
				
			||||||
		ObjectWriter writer = (jsonView != null ?
 | 
					 | 
				
			||||||
				objectMapper().writerWithView(jsonView) : objectMapper().writer());
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		if (javaType.isContainerType()) {
 | 
					 | 
				
			||||||
			writer = writer.forType(javaType);
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		if (MediaType.TEXT_EVENT_STREAM.isCompatibleWith(mimeType) &&
 | 
					 | 
				
			||||||
				writer.getConfig().isEnabled(SerializationFeature.INDENT_OUTPUT)) {
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
			writer = writer.with(this.ssePrettyPrinter);
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		DataBuffer buffer = bufferFactory.allocateBuffer();
 | 
					 | 
				
			||||||
		OutputStream outputStream = buffer.asOutputStream();
 | 
					 | 
				
			||||||
		try {
 | 
					 | 
				
			||||||
			writer.writeValue(outputStream, value);
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
		catch (InvalidDefinitionException ex) {
 | 
					 | 
				
			||||||
			throw new CodecException("Type definition error: " + ex.getType(), ex);
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
		catch (JsonProcessingException ex) {
 | 
					 | 
				
			||||||
			throw new EncodingException("JSON encoding error: " + ex.getOriginalMessage(), ex);
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
		catch (IOException ex) {
 | 
					 | 
				
			||||||
			throw new IllegalStateException("Unexpected I/O error while writing to data buffer", ex);
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		return buffer;
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	// HttpMessageEncoder...
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	@Override
 | 
					 | 
				
			||||||
	public List<MediaType> getStreamingMediaTypes() {
 | 
					 | 
				
			||||||
		return Collections.unmodifiableList(this.streamingMediaTypes);
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	@Override
 | 
					 | 
				
			||||||
	public Map<String, Object> getEncodeHints(@Nullable ResolvableType actualType, ResolvableType elementType,
 | 
					 | 
				
			||||||
			@Nullable MediaType mediaType, ServerHttpRequest request, ServerHttpResponse response) {
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		return (actualType != null ? getHints(actualType) : Collections.emptyMap());
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	@Override
 | 
					 | 
				
			||||||
	protected <A extends Annotation> A getAnnotation(MethodParameter parameter, Class<A> annotType) {
 | 
					 | 
				
			||||||
		return parameter.getMethodAnnotation(annotType);
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,56 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * 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.http.codec.json;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import java.util.Arrays;
 | 
				
			||||||
 | 
					import java.util.List;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.databind.ObjectMapper;
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.dataformat.smile.SmileFactory;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import org.springframework.http.MediaType;
 | 
				
			||||||
 | 
					import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
 | 
				
			||||||
 | 
					import org.springframework.util.Assert;
 | 
				
			||||||
 | 
					import org.springframework.util.MimeType;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * Decode a byte stream into Smile and convert to Object's with Jackson 2.9.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * @author Sebastien Deleuze
 | 
				
			||||||
 | 
					 * @author Rossen Stoyanchev
 | 
				
			||||||
 | 
					 * @since 5.0
 | 
				
			||||||
 | 
					 * @see Jackson2JsonEncoder
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					public class Jackson2SmileDecoder extends AbstractJackson2Decoder {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						private static final MimeType SMILE_MIME_TYPE = new MediaType("application", "x-jackson-smile");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						public Jackson2SmileDecoder() {
 | 
				
			||||||
 | 
							this(Jackson2ObjectMapperBuilder.smile().build(), SMILE_MIME_TYPE);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						public Jackson2SmileDecoder(ObjectMapper mapper, MimeType... mimeTypes) {
 | 
				
			||||||
 | 
							super(mapper, mimeTypes);
 | 
				
			||||||
 | 
							Assert.isAssignable(SmileFactory.class, mapper.getFactory().getClass());
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Override
 | 
				
			||||||
 | 
						public List<MimeType> getDecodableMimeTypes() {
 | 
				
			||||||
 | 
							return Arrays.asList(SMILE_MIME_TYPE);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,56 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * 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.http.codec.json;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import java.util.Arrays;
 | 
				
			||||||
 | 
					import java.util.List;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.databind.ObjectMapper;
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.dataformat.smile.SmileFactory;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import org.springframework.http.MediaType;
 | 
				
			||||||
 | 
					import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
 | 
				
			||||||
 | 
					import org.springframework.util.Assert;
 | 
				
			||||||
 | 
					import org.springframework.util.MimeType;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * Encode from an {@code Object} stream to a byte stream of Smile objects using Jackson 2.9.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * @author Sebastien Deleuze
 | 
				
			||||||
 | 
					 * @since 5.0
 | 
				
			||||||
 | 
					 * @see Jackson2SmileDecoder
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					public class Jackson2SmileEncoder extends AbstractJackson2Encoder {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						private static final MimeType SMILE_MIME_TYPE = new MediaType("application", "x-jackson-smile");
 | 
				
			||||||
 | 
						
 | 
				
			||||||
 | 
						
 | 
				
			||||||
 | 
						public Jackson2SmileEncoder() {
 | 
				
			||||||
 | 
							this(Jackson2ObjectMapperBuilder.smile().build(), new MediaType("application", "x-jackson-smile"));
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						public Jackson2SmileEncoder(ObjectMapper mapper, MimeType... mimeTypes) {
 | 
				
			||||||
 | 
							super(mapper, mimeTypes);
 | 
				
			||||||
 | 
							Assert.isAssignable(SmileFactory.class, mapper.getFactory().getClass());
 | 
				
			||||||
 | 
							this.streamingMediaTypes.add(new MediaType("application", "stream+x-jackson-smile"));
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Override
 | 
				
			||||||
 | 
						public List<MimeType> getEncodableMimeTypes() {
 | 
				
			||||||
 | 
							return Arrays.asList(SMILE_MIME_TYPE);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -40,6 +40,8 @@ import org.springframework.core.io.buffer.DefaultDataBufferFactory;
 | 
				
			||||||
import org.springframework.http.MediaType;
 | 
					import org.springframework.http.MediaType;
 | 
				
			||||||
import org.springframework.http.codec.json.Jackson2JsonDecoder;
 | 
					import org.springframework.http.codec.json.Jackson2JsonDecoder;
 | 
				
			||||||
import org.springframework.http.codec.json.Jackson2JsonEncoder;
 | 
					import org.springframework.http.codec.json.Jackson2JsonEncoder;
 | 
				
			||||||
 | 
					import org.springframework.http.codec.json.Jackson2SmileDecoder;
 | 
				
			||||||
 | 
					import org.springframework.http.codec.json.Jackson2SmileEncoder;
 | 
				
			||||||
import org.springframework.http.codec.multipart.MultipartHttpMessageWriter;
 | 
					import org.springframework.http.codec.multipart.MultipartHttpMessageWriter;
 | 
				
			||||||
import org.springframework.http.codec.xml.Jaxb2XmlDecoder;
 | 
					import org.springframework.http.codec.xml.Jaxb2XmlDecoder;
 | 
				
			||||||
import org.springframework.http.codec.xml.Jaxb2XmlEncoder;
 | 
					import org.springframework.http.codec.xml.Jaxb2XmlEncoder;
 | 
				
			||||||
| 
						 | 
					@ -62,7 +64,7 @@ public class ClientCodecConfigurerTests {
 | 
				
			||||||
	@Test
 | 
						@Test
 | 
				
			||||||
	public void defaultReaders() throws Exception {
 | 
						public void defaultReaders() throws Exception {
 | 
				
			||||||
		List<HttpMessageReader<?>> readers = this.configurer.getReaders();
 | 
							List<HttpMessageReader<?>> readers = this.configurer.getReaders();
 | 
				
			||||||
		assertEquals(9, readers.size());
 | 
							assertEquals(10, readers.size());
 | 
				
			||||||
		assertEquals(ByteArrayDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(ByteArrayDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
		assertEquals(ByteBufferDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(ByteBufferDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
		assertEquals(DataBufferDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(DataBufferDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
| 
						 | 
					@ -70,6 +72,7 @@ public class ClientCodecConfigurerTests {
 | 
				
			||||||
		assertStringDecoder(getNextDecoder(readers), true);
 | 
							assertStringDecoder(getNextDecoder(readers), true);
 | 
				
			||||||
		assertEquals(Jaxb2XmlDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(Jaxb2XmlDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
		assertEquals(Jackson2JsonDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(Jackson2JsonDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
 | 
							assertEquals(Jackson2SmileDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
		assertSseReader(readers);
 | 
							assertSseReader(readers);
 | 
				
			||||||
		assertStringDecoder(getNextDecoder(readers), false);
 | 
							assertStringDecoder(getNextDecoder(readers), false);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
| 
						 | 
					@ -77,7 +80,7 @@ public class ClientCodecConfigurerTests {
 | 
				
			||||||
	@Test
 | 
						@Test
 | 
				
			||||||
	public void defaultWriters() throws Exception {
 | 
						public void defaultWriters() throws Exception {
 | 
				
			||||||
		List<HttpMessageWriter<?>> writers = this.configurer.getWriters();
 | 
							List<HttpMessageWriter<?>> writers = this.configurer.getWriters();
 | 
				
			||||||
		assertEquals(10, writers.size());
 | 
							assertEquals(11, writers.size());
 | 
				
			||||||
		assertEquals(ByteArrayEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(ByteArrayEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
		assertEquals(ByteBufferEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(ByteBufferEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
		assertEquals(DataBufferEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(DataBufferEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
| 
						 | 
					@ -87,6 +90,7 @@ public class ClientCodecConfigurerTests {
 | 
				
			||||||
		assertEquals(MultipartHttpMessageWriter.class, writers.get(this.index.getAndIncrement()).getClass());
 | 
							assertEquals(MultipartHttpMessageWriter.class, writers.get(this.index.getAndIncrement()).getClass());
 | 
				
			||||||
		assertEquals(Jaxb2XmlEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(Jaxb2XmlEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
		assertEquals(Jackson2JsonEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(Jackson2JsonEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
 | 
							assertEquals(Jackson2SmileEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
		assertStringEncoder(getNextEncoder(writers), false);
 | 
							assertStringEncoder(getNextEncoder(writers), false);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -94,7 +98,7 @@ public class ClientCodecConfigurerTests {
 | 
				
			||||||
	public void jackson2EncoderOverride() throws Exception {
 | 
						public void jackson2EncoderOverride() throws Exception {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		Jackson2JsonDecoder decoder = new Jackson2JsonDecoder();
 | 
							Jackson2JsonDecoder decoder = new Jackson2JsonDecoder();
 | 
				
			||||||
		this.configurer.defaultCodecs().jackson2Decoder(decoder);
 | 
							this.configurer.defaultCodecs().jackson2JsonDecoder(decoder);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		assertSame(decoder, this.configurer.getReaders().stream()
 | 
							assertSame(decoder, this.configurer.getReaders().stream()
 | 
				
			||||||
				.filter(reader -> ServerSentEventHttpMessageReader.class.equals(reader.getClass()))
 | 
									.filter(reader -> ServerSentEventHttpMessageReader.class.equals(reader.getClass()))
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -35,6 +35,8 @@ import org.springframework.core.codec.StringDecoder;
 | 
				
			||||||
import org.springframework.http.MediaType;
 | 
					import org.springframework.http.MediaType;
 | 
				
			||||||
import org.springframework.http.codec.json.Jackson2JsonDecoder;
 | 
					import org.springframework.http.codec.json.Jackson2JsonDecoder;
 | 
				
			||||||
import org.springframework.http.codec.json.Jackson2JsonEncoder;
 | 
					import org.springframework.http.codec.json.Jackson2JsonEncoder;
 | 
				
			||||||
 | 
					import org.springframework.http.codec.json.Jackson2SmileDecoder;
 | 
				
			||||||
 | 
					import org.springframework.http.codec.json.Jackson2SmileEncoder;
 | 
				
			||||||
import org.springframework.http.codec.xml.Jaxb2XmlDecoder;
 | 
					import org.springframework.http.codec.xml.Jaxb2XmlDecoder;
 | 
				
			||||||
import org.springframework.http.codec.xml.Jaxb2XmlEncoder;
 | 
					import org.springframework.http.codec.xml.Jaxb2XmlEncoder;
 | 
				
			||||||
import org.springframework.util.MimeTypeUtils;
 | 
					import org.springframework.util.MimeTypeUtils;
 | 
				
			||||||
| 
						 | 
					@ -60,7 +62,7 @@ public class CodecConfigurerTests {
 | 
				
			||||||
	@Test
 | 
						@Test
 | 
				
			||||||
	public void defaultReaders() throws Exception {
 | 
						public void defaultReaders() throws Exception {
 | 
				
			||||||
		List<HttpMessageReader<?>> readers = this.configurer.getReaders();
 | 
							List<HttpMessageReader<?>> readers = this.configurer.getReaders();
 | 
				
			||||||
		assertEquals(8, readers.size());
 | 
							assertEquals(9, readers.size());
 | 
				
			||||||
		assertEquals(ByteArrayDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(ByteArrayDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
		assertEquals(ByteBufferDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(ByteBufferDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
		assertEquals(DataBufferDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(DataBufferDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
| 
						 | 
					@ -68,13 +70,14 @@ public class CodecConfigurerTests {
 | 
				
			||||||
		assertStringDecoder(getNextDecoder(readers), true);
 | 
							assertStringDecoder(getNextDecoder(readers), true);
 | 
				
			||||||
		assertEquals(Jaxb2XmlDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(Jaxb2XmlDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
		assertEquals(Jackson2JsonDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(Jackson2JsonDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
 | 
							assertEquals(Jackson2SmileDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
		assertStringDecoder(getNextDecoder(readers), false);
 | 
							assertStringDecoder(getNextDecoder(readers), false);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	@Test
 | 
						@Test
 | 
				
			||||||
	public void defaultWriters() throws Exception {
 | 
						public void defaultWriters() throws Exception {
 | 
				
			||||||
		List<HttpMessageWriter<?>> writers = this.configurer.getWriters();
 | 
							List<HttpMessageWriter<?>> writers = this.configurer.getWriters();
 | 
				
			||||||
		assertEquals(8, writers.size());
 | 
							assertEquals(9, writers.size());
 | 
				
			||||||
		assertEquals(ByteArrayEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(ByteArrayEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
		assertEquals(ByteBufferEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(ByteBufferEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
		assertEquals(DataBufferEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(DataBufferEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
| 
						 | 
					@ -82,6 +85,7 @@ public class CodecConfigurerTests {
 | 
				
			||||||
		assertStringEncoder(getNextEncoder(writers), true);
 | 
							assertStringEncoder(getNextEncoder(writers), true);
 | 
				
			||||||
		assertEquals(Jaxb2XmlEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(Jaxb2XmlEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
		assertEquals(Jackson2JsonEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(Jackson2JsonEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
 | 
							assertEquals(Jackson2SmileEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
		assertStringEncoder(getNextEncoder(writers), false);
 | 
							assertStringEncoder(getNextEncoder(writers), false);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -108,7 +112,7 @@ public class CodecConfigurerTests {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		List<HttpMessageReader<?>> readers = this.configurer.getReaders();
 | 
							List<HttpMessageReader<?>> readers = this.configurer.getReaders();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		assertEquals(12, readers.size());
 | 
							assertEquals(13, readers.size());
 | 
				
			||||||
		assertEquals(ByteArrayDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(ByteArrayDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
		assertEquals(ByteBufferDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(ByteBufferDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
		assertEquals(DataBufferDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(DataBufferDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
| 
						 | 
					@ -118,6 +122,7 @@ public class CodecConfigurerTests {
 | 
				
			||||||
		assertSame(customReader1, readers.get(this.index.getAndIncrement()));
 | 
							assertSame(customReader1, readers.get(this.index.getAndIncrement()));
 | 
				
			||||||
		assertEquals(Jaxb2XmlDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(Jaxb2XmlDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
		assertEquals(Jackson2JsonDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(Jackson2JsonDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
 | 
							assertEquals(Jackson2SmileDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
		assertSame(customDecoder2, getNextDecoder(readers));
 | 
							assertSame(customDecoder2, getNextDecoder(readers));
 | 
				
			||||||
		assertSame(customReader2, readers.get(this.index.getAndIncrement()));
 | 
							assertSame(customReader2, readers.get(this.index.getAndIncrement()));
 | 
				
			||||||
		assertEquals(StringDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(StringDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
| 
						 | 
					@ -146,7 +151,7 @@ public class CodecConfigurerTests {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		List<HttpMessageWriter<?>> writers = this.configurer.getWriters();
 | 
							List<HttpMessageWriter<?>> writers = this.configurer.getWriters();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		assertEquals(12, writers.size());
 | 
							assertEquals(13, writers.size());
 | 
				
			||||||
		assertEquals(ByteArrayEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(ByteArrayEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
		assertEquals(ByteBufferEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(ByteBufferEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
		assertEquals(DataBufferEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(DataBufferEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
| 
						 | 
					@ -156,6 +161,7 @@ public class CodecConfigurerTests {
 | 
				
			||||||
		assertSame(customWriter1, writers.get(this.index.getAndIncrement()));
 | 
							assertSame(customWriter1, writers.get(this.index.getAndIncrement()));
 | 
				
			||||||
		assertEquals(Jaxb2XmlEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(Jaxb2XmlEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
		assertEquals(Jackson2JsonEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(Jackson2JsonEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
 | 
							assertEquals(Jackson2SmileEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
		assertSame(customEncoder2, getNextEncoder(writers));
 | 
							assertSame(customEncoder2, getNextEncoder(writers));
 | 
				
			||||||
		assertSame(customWriter2, writers.get(this.index.getAndIncrement()));
 | 
							assertSame(customWriter2, writers.get(this.index.getAndIncrement()));
 | 
				
			||||||
		assertEquals(CharSequenceEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(CharSequenceEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
| 
						 | 
					@ -229,7 +235,7 @@ public class CodecConfigurerTests {
 | 
				
			||||||
	public void jackson2DecoderOverride() throws Exception {
 | 
						public void jackson2DecoderOverride() throws Exception {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		Jackson2JsonDecoder decoder = new Jackson2JsonDecoder();
 | 
							Jackson2JsonDecoder decoder = new Jackson2JsonDecoder();
 | 
				
			||||||
		this.configurer.defaultCodecs().jackson2Decoder(decoder);
 | 
							this.configurer.defaultCodecs().jackson2JsonDecoder(decoder);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		assertSame(decoder, this.configurer.getReaders().stream()
 | 
							assertSame(decoder, this.configurer.getReaders().stream()
 | 
				
			||||||
				.filter(writer -> writer instanceof DecoderHttpMessageReader)
 | 
									.filter(writer -> writer instanceof DecoderHttpMessageReader)
 | 
				
			||||||
| 
						 | 
					@ -243,7 +249,7 @@ public class CodecConfigurerTests {
 | 
				
			||||||
	public void jackson2EncoderOverride() throws Exception {
 | 
						public void jackson2EncoderOverride() throws Exception {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		Jackson2JsonEncoder encoder = new Jackson2JsonEncoder();
 | 
							Jackson2JsonEncoder encoder = new Jackson2JsonEncoder();
 | 
				
			||||||
		this.configurer.defaultCodecs().jackson2Encoder(encoder);
 | 
							this.configurer.defaultCodecs().jackson2JsonEncoder(encoder);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		assertSame(encoder, this.configurer.getWriters().stream()
 | 
							assertSame(encoder, this.configurer.getWriters().stream()
 | 
				
			||||||
				.filter(writer -> writer instanceof EncoderHttpMessageWriter)
 | 
									.filter(writer -> writer instanceof EncoderHttpMessageWriter)
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -41,6 +41,8 @@ import org.springframework.core.io.buffer.DefaultDataBufferFactory;
 | 
				
			||||||
import org.springframework.http.MediaType;
 | 
					import org.springframework.http.MediaType;
 | 
				
			||||||
import org.springframework.http.codec.json.Jackson2JsonDecoder;
 | 
					import org.springframework.http.codec.json.Jackson2JsonDecoder;
 | 
				
			||||||
import org.springframework.http.codec.json.Jackson2JsonEncoder;
 | 
					import org.springframework.http.codec.json.Jackson2JsonEncoder;
 | 
				
			||||||
 | 
					import org.springframework.http.codec.json.Jackson2SmileDecoder;
 | 
				
			||||||
 | 
					import org.springframework.http.codec.json.Jackson2SmileEncoder;
 | 
				
			||||||
import org.springframework.http.codec.multipart.MultipartHttpMessageReader;
 | 
					import org.springframework.http.codec.multipart.MultipartHttpMessageReader;
 | 
				
			||||||
import org.springframework.http.codec.multipart.SynchronossPartHttpMessageReader;
 | 
					import org.springframework.http.codec.multipart.SynchronossPartHttpMessageReader;
 | 
				
			||||||
import org.springframework.http.codec.xml.Jaxb2XmlDecoder;
 | 
					import org.springframework.http.codec.xml.Jaxb2XmlDecoder;
 | 
				
			||||||
| 
						 | 
					@ -64,7 +66,7 @@ public class ServerCodecConfigurerTests {
 | 
				
			||||||
	@Test
 | 
						@Test
 | 
				
			||||||
	public void defaultReaders() throws Exception {
 | 
						public void defaultReaders() throws Exception {
 | 
				
			||||||
		List<HttpMessageReader<?>> readers = this.configurer.getReaders();
 | 
							List<HttpMessageReader<?>> readers = this.configurer.getReaders();
 | 
				
			||||||
		assertEquals(11, readers.size());
 | 
							assertEquals(12, readers.size());
 | 
				
			||||||
		assertEquals(ByteArrayDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(ByteArrayDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
		assertEquals(ByteBufferDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(ByteBufferDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
		assertEquals(DataBufferDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(DataBufferDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
| 
						 | 
					@ -75,13 +77,14 @@ public class ServerCodecConfigurerTests {
 | 
				
			||||||
		assertEquals(MultipartHttpMessageReader.class, readers.get(this.index.getAndIncrement()).getClass());
 | 
							assertEquals(MultipartHttpMessageReader.class, readers.get(this.index.getAndIncrement()).getClass());
 | 
				
			||||||
		assertEquals(Jaxb2XmlDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(Jaxb2XmlDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
		assertEquals(Jackson2JsonDecoder.class, getNextDecoder(readers).getClass());
 | 
							assertEquals(Jackson2JsonDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
 | 
							assertEquals(Jackson2SmileDecoder.class, getNextDecoder(readers).getClass());
 | 
				
			||||||
		assertStringDecoder(getNextDecoder(readers), false);
 | 
							assertStringDecoder(getNextDecoder(readers), false);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	@Test
 | 
						@Test
 | 
				
			||||||
	public void defaultWriters() throws Exception {
 | 
						public void defaultWriters() throws Exception {
 | 
				
			||||||
		List<HttpMessageWriter<?>> writers = this.configurer.getWriters();
 | 
							List<HttpMessageWriter<?>> writers = this.configurer.getWriters();
 | 
				
			||||||
		assertEquals(9, writers.size());
 | 
							assertEquals(10, writers.size());
 | 
				
			||||||
		assertEquals(ByteArrayEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(ByteArrayEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
		assertEquals(ByteBufferEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(ByteBufferEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
		assertEquals(DataBufferEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(DataBufferEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
| 
						 | 
					@ -89,6 +92,7 @@ public class ServerCodecConfigurerTests {
 | 
				
			||||||
		assertStringEncoder(getNextEncoder(writers), true);
 | 
							assertStringEncoder(getNextEncoder(writers), true);
 | 
				
			||||||
		assertEquals(Jaxb2XmlEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(Jaxb2XmlEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
		assertEquals(Jackson2JsonEncoder.class, getNextEncoder(writers).getClass());
 | 
							assertEquals(Jackson2JsonEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
 | 
							assertEquals(Jackson2SmileEncoder.class, getNextEncoder(writers).getClass());
 | 
				
			||||||
		assertSseWriter(writers);
 | 
							assertSseWriter(writers);
 | 
				
			||||||
		assertStringEncoder(getNextEncoder(writers), false);
 | 
							assertStringEncoder(getNextEncoder(writers), false);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
| 
						 | 
					@ -97,7 +101,7 @@ public class ServerCodecConfigurerTests {
 | 
				
			||||||
	public void jackson2EncoderOverride() throws Exception {
 | 
						public void jackson2EncoderOverride() throws Exception {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		Jackson2JsonEncoder encoder = new Jackson2JsonEncoder();
 | 
							Jackson2JsonEncoder encoder = new Jackson2JsonEncoder();
 | 
				
			||||||
		this.configurer.defaultCodecs().jackson2Encoder(encoder);
 | 
							this.configurer.defaultCodecs().jackson2JsonEncoder(encoder);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		assertSame(encoder, this.configurer.getWriters().stream()
 | 
							assertSame(encoder, this.configurer.getWriters().stream()
 | 
				
			||||||
				.filter(writer -> ServerSentEventHttpMessageWriter.class.equals(writer.getClass()))
 | 
									.filter(writer -> ServerSentEventHttpMessageWriter.class.equals(writer.getClass()))
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,118 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * 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.http.codec.json;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import java.util.List;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.databind.ObjectMapper;
 | 
				
			||||||
 | 
					import org.junit.Test;
 | 
				
			||||||
 | 
					import reactor.core.publisher.Flux;
 | 
				
			||||||
 | 
					import reactor.core.publisher.Mono;
 | 
				
			||||||
 | 
					import reactor.test.StepVerifier;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import org.springframework.core.ResolvableType;
 | 
				
			||||||
 | 
					import org.springframework.core.codec.CodecException;
 | 
				
			||||||
 | 
					import org.springframework.core.io.buffer.AbstractDataBufferAllocatingTestCase;
 | 
				
			||||||
 | 
					import org.springframework.core.io.buffer.DataBuffer;
 | 
				
			||||||
 | 
					import org.springframework.http.codec.Pojo;
 | 
				
			||||||
 | 
					import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
 | 
				
			||||||
 | 
					import org.springframework.util.MimeType;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import static java.util.Arrays.asList;
 | 
				
			||||||
 | 
					import static java.util.Collections.emptyMap;
 | 
				
			||||||
 | 
					import static org.junit.Assert.assertFalse;
 | 
				
			||||||
 | 
					import static org.junit.Assert.assertTrue;
 | 
				
			||||||
 | 
					import static org.springframework.core.ResolvableType.forClass;
 | 
				
			||||||
 | 
					import static org.springframework.http.MediaType.APPLICATION_JSON;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * Unit tests for {@link Jackson2SmileDecoder}.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * @author Sebastien Deleuze
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					public class Jackson2SmileDecoderTests extends AbstractDataBufferAllocatingTestCase {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						private final static MimeType SMILE_MIME_TYPE = new MimeType("application", "x-jackson-smile");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						private final Jackson2SmileDecoder decoder = new Jackson2SmileDecoder();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Test
 | 
				
			||||||
 | 
						public void canDecode() {
 | 
				
			||||||
 | 
							assertTrue(decoder.canDecode(forClass(Pojo.class), SMILE_MIME_TYPE));
 | 
				
			||||||
 | 
							assertTrue(decoder.canDecode(forClass(Pojo.class), null));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							assertFalse(decoder.canDecode(forClass(String.class), null));
 | 
				
			||||||
 | 
							assertFalse(decoder.canDecode(forClass(Pojo.class), APPLICATION_JSON));
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Test
 | 
				
			||||||
 | 
						public void decodePojo() throws Exception {
 | 
				
			||||||
 | 
							ObjectMapper mapper = Jackson2ObjectMapperBuilder.smile().build();
 | 
				
			||||||
 | 
							Pojo pojo = new Pojo("foo", "bar");
 | 
				
			||||||
 | 
							byte[] serializedPojo = mapper.writer().writeValueAsBytes(pojo);
 | 
				
			||||||
 | 
							
 | 
				
			||||||
 | 
							Flux<DataBuffer> source = Flux.just(this.bufferFactory.wrap(serializedPojo));
 | 
				
			||||||
 | 
							ResolvableType elementType = forClass(Pojo.class);
 | 
				
			||||||
 | 
							Flux<Object> flux = decoder.decode(source, elementType, null, emptyMap());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							StepVerifier.create(flux)
 | 
				
			||||||
 | 
									.expectNext(pojo)
 | 
				
			||||||
 | 
									.verifyComplete();
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Test
 | 
				
			||||||
 | 
						public void decodePojoWithError() throws Exception {
 | 
				
			||||||
 | 
							Flux<DataBuffer> source = Flux.just(stringBuffer("123"));
 | 
				
			||||||
 | 
							ResolvableType elementType = forClass(Pojo.class);
 | 
				
			||||||
 | 
							Flux<Object> flux = decoder.decode(source, elementType, null, emptyMap());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							StepVerifier.create(flux).verifyError(CodecException.class);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Test
 | 
				
			||||||
 | 
						public void decodeToList() throws Exception {
 | 
				
			||||||
 | 
							ObjectMapper mapper = Jackson2ObjectMapperBuilder.smile().build();
 | 
				
			||||||
 | 
							List<Pojo> list = asList(new Pojo("f1", "b1"), new Pojo("f2", "b2"));
 | 
				
			||||||
 | 
							byte[] serializedList = mapper.writer().writeValueAsBytes(list);
 | 
				
			||||||
 | 
							Flux<DataBuffer> source = Flux.just(this.bufferFactory.wrap(serializedList));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							ResolvableType elementType = ResolvableType.forClassWithGenerics(List.class, Pojo.class);
 | 
				
			||||||
 | 
							Mono<Object> mono = decoder.decodeToMono(source, elementType, null, emptyMap());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							StepVerifier.create(mono)
 | 
				
			||||||
 | 
									.expectNext(list)
 | 
				
			||||||
 | 
									.expectComplete()
 | 
				
			||||||
 | 
									.verify();
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Test
 | 
				
			||||||
 | 
						public void decodeToFlux() throws Exception {
 | 
				
			||||||
 | 
							ObjectMapper mapper = Jackson2ObjectMapperBuilder.smile().build();
 | 
				
			||||||
 | 
							List<Pojo> list = asList(new Pojo("f1", "b1"), new Pojo("f2", "b2"));
 | 
				
			||||||
 | 
							byte[] serializedList = mapper.writer().writeValueAsBytes(list);
 | 
				
			||||||
 | 
							Flux<DataBuffer> source = Flux.just(this.bufferFactory.wrap(serializedList));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							ResolvableType elementType = forClass(Pojo.class);
 | 
				
			||||||
 | 
							Flux<Object> flux = decoder.decode(source, elementType, null, emptyMap());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							StepVerifier.create(flux)
 | 
				
			||||||
 | 
									.expectNext(new Pojo("f1", "b1"))
 | 
				
			||||||
 | 
									.expectNext(new Pojo("f2", "b2"))
 | 
				
			||||||
 | 
									.verifyComplete();
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,126 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * 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.http.codec.json;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import java.io.IOException;
 | 
				
			||||||
 | 
					import java.io.UncheckedIOException;
 | 
				
			||||||
 | 
					import java.util.Arrays;
 | 
				
			||||||
 | 
					import java.util.List;
 | 
				
			||||||
 | 
					import java.util.Map;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.annotation.JsonTypeInfo;
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.annotation.JsonTypeName;
 | 
				
			||||||
 | 
					import com.fasterxml.jackson.databind.ObjectMapper;
 | 
				
			||||||
 | 
					import org.junit.Test;
 | 
				
			||||||
 | 
					import reactor.core.publisher.Flux;
 | 
				
			||||||
 | 
					import reactor.core.publisher.Mono;
 | 
				
			||||||
 | 
					import reactor.test.StepVerifier;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import org.springframework.core.ResolvableType;
 | 
				
			||||||
 | 
					import org.springframework.core.io.buffer.AbstractDataBufferAllocatingTestCase;
 | 
				
			||||||
 | 
					import org.springframework.core.io.buffer.DataBuffer;
 | 
				
			||||||
 | 
					import org.springframework.core.io.buffer.DataBufferUtils;
 | 
				
			||||||
 | 
					import org.springframework.core.io.buffer.support.DataBufferTestUtils;
 | 
				
			||||||
 | 
					import org.springframework.http.MediaType;
 | 
				
			||||||
 | 
					import org.springframework.http.codec.Pojo;
 | 
				
			||||||
 | 
					import org.springframework.http.codec.ServerSentEvent;
 | 
				
			||||||
 | 
					import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
 | 
				
			||||||
 | 
					import org.springframework.util.MimeType;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import static java.util.Collections.emptyMap;
 | 
				
			||||||
 | 
					import static org.junit.Assert.assertFalse;
 | 
				
			||||||
 | 
					import static org.junit.Assert.assertTrue;
 | 
				
			||||||
 | 
					import static org.springframework.http.MediaType.APPLICATION_XML;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * Unit tests for {@link Jackson2SmileEncoder}.
 | 
				
			||||||
 | 
					 * 
 | 
				
			||||||
 | 
					 * @author Sebastien Deleuze
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					public class Jackson2SmileEncoderTests extends AbstractDataBufferAllocatingTestCase {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						private final static MimeType SMILE_MIME_TYPE = new MimeType("application", "x-jackson-smile");
 | 
				
			||||||
 | 
						
 | 
				
			||||||
 | 
						private final Jackson2SmileEncoder encoder = new Jackson2SmileEncoder();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Test
 | 
				
			||||||
 | 
						public void canEncode() {
 | 
				
			||||||
 | 
							ResolvableType pojoType = ResolvableType.forClass(Pojo.class);
 | 
				
			||||||
 | 
							assertTrue(this.encoder.canEncode(pojoType, SMILE_MIME_TYPE));
 | 
				
			||||||
 | 
							assertTrue(this.encoder.canEncode(pojoType, null));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							// SPR-15464
 | 
				
			||||||
 | 
							assertTrue(this.encoder.canEncode(ResolvableType.NONE, null));
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Test
 | 
				
			||||||
 | 
						public void canNotEncode() {
 | 
				
			||||||
 | 
							assertFalse(this.encoder.canEncode(ResolvableType.forClass(String.class), null));
 | 
				
			||||||
 | 
							assertFalse(this.encoder.canEncode(ResolvableType.forClass(Pojo.class), APPLICATION_XML));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							ResolvableType sseType = ResolvableType.forClass(ServerSentEvent.class);
 | 
				
			||||||
 | 
							assertFalse(this.encoder.canEncode(sseType, SMILE_MIME_TYPE));
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Test
 | 
				
			||||||
 | 
						public void encode() throws Exception {
 | 
				
			||||||
 | 
							Flux<Pojo> source = Flux.just(
 | 
				
			||||||
 | 
									new Pojo("foo", "bar"),
 | 
				
			||||||
 | 
									new Pojo("foofoo", "barbar"),
 | 
				
			||||||
 | 
									new Pojo("foofoofoo", "barbarbar")
 | 
				
			||||||
 | 
							);
 | 
				
			||||||
 | 
							ResolvableType type = ResolvableType.forClass(Pojo.class);
 | 
				
			||||||
 | 
							Flux<DataBuffer> output = this.encoder.encode(source, this.bufferFactory, type, null, emptyMap());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							ObjectMapper mapper = Jackson2ObjectMapperBuilder.smile().build();
 | 
				
			||||||
 | 
							StepVerifier.create(output)
 | 
				
			||||||
 | 
									.consumeNextWith(dataBuffer -> readPojo(mapper, List.class, dataBuffer))
 | 
				
			||||||
 | 
									.verifyComplete();
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@Test
 | 
				
			||||||
 | 
						public void encodeAsStream() throws Exception {
 | 
				
			||||||
 | 
							Flux<Pojo> source = Flux.just(
 | 
				
			||||||
 | 
									new Pojo("foo", "bar"),
 | 
				
			||||||
 | 
									new Pojo("foofoo", "barbar"),
 | 
				
			||||||
 | 
									new Pojo("foofoofoo", "barbarbar")
 | 
				
			||||||
 | 
							);
 | 
				
			||||||
 | 
							ResolvableType type = ResolvableType.forClass(Pojo.class);
 | 
				
			||||||
 | 
							MediaType mediaType = new MediaType("application", "stream+x-jackson-smile");
 | 
				
			||||||
 | 
							Flux<DataBuffer> output = this.encoder.encode(source, this.bufferFactory, type, mediaType, emptyMap());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							ObjectMapper mapper = Jackson2ObjectMapperBuilder.smile().build();
 | 
				
			||||||
 | 
							StepVerifier.create(output)
 | 
				
			||||||
 | 
									.consumeNextWith(dataBuffer -> readPojo(mapper, Pojo.class, dataBuffer))
 | 
				
			||||||
 | 
									.consumeNextWith(dataBuffer -> readPojo(mapper, Pojo.class, dataBuffer))
 | 
				
			||||||
 | 
									.consumeNextWith(dataBuffer -> readPojo(mapper, Pojo.class, dataBuffer))
 | 
				
			||||||
 | 
									.verifyComplete();
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						
 | 
				
			||||||
 | 
						public <T> T readPojo(ObjectMapper mapper, Class<T> valueType, DataBuffer dataBuffer) {
 | 
				
			||||||
 | 
							try {
 | 
				
			||||||
 | 
								T value = mapper.reader().forType(valueType).readValue(DataBufferTestUtils.dumpBytes(dataBuffer));
 | 
				
			||||||
 | 
								DataBufferUtils.release(dataBuffer);
 | 
				
			||||||
 | 
								return value;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							catch (IOException ex) {
 | 
				
			||||||
 | 
								throw new UncheckedIOException(ex);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -97,7 +97,7 @@ public class DelegatingWebFluxConfigurationTests {
 | 
				
			||||||
		verify(webFluxConfigurer).configureArgumentResolvers(any());
 | 
							verify(webFluxConfigurer).configureArgumentResolvers(any());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		assertSame(formatterRegistry.getValue(), initializerConversionService);
 | 
							assertSame(formatterRegistry.getValue(), initializerConversionService);
 | 
				
			||||||
		assertEquals(11, codecsConfigurer.getValue().getReaders().size());
 | 
							assertEquals(12, codecsConfigurer.getValue().getReaders().size());
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	@Test
 | 
						@Test
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -138,7 +138,7 @@ public class WebFluxConfigurationSupportTests {
 | 
				
			||||||
		assertNotNull(adapter);
 | 
							assertNotNull(adapter);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		List<HttpMessageReader<?>> readers = adapter.getMessageCodecConfigurer().getReaders();
 | 
							List<HttpMessageReader<?>> readers = adapter.getMessageCodecConfigurer().getReaders();
 | 
				
			||||||
		assertEquals(11, readers.size());
 | 
							assertEquals(12, readers.size());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		assertHasMessageReader(readers, forClass(byte[].class), APPLICATION_OCTET_STREAM);
 | 
							assertHasMessageReader(readers, forClass(byte[].class), APPLICATION_OCTET_STREAM);
 | 
				
			||||||
		assertHasMessageReader(readers, forClass(ByteBuffer.class), APPLICATION_OCTET_STREAM);
 | 
							assertHasMessageReader(readers, forClass(ByteBuffer.class), APPLICATION_OCTET_STREAM);
 | 
				
			||||||
| 
						 | 
					@ -147,6 +147,7 @@ public class WebFluxConfigurationSupportTests {
 | 
				
			||||||
		assertHasMessageReader(readers, forClassWithGenerics(MultiValueMap.class, String.class, String.class), APPLICATION_FORM_URLENCODED);
 | 
							assertHasMessageReader(readers, forClassWithGenerics(MultiValueMap.class, String.class, String.class), APPLICATION_FORM_URLENCODED);
 | 
				
			||||||
		assertHasMessageReader(readers, forClass(TestBean.class), APPLICATION_XML);
 | 
							assertHasMessageReader(readers, forClass(TestBean.class), APPLICATION_XML);
 | 
				
			||||||
		assertHasMessageReader(readers, forClass(TestBean.class), APPLICATION_JSON);
 | 
							assertHasMessageReader(readers, forClass(TestBean.class), APPLICATION_JSON);
 | 
				
			||||||
 | 
							assertHasMessageReader(readers, forClass(TestBean.class), new MediaType("application", "x-jackson-smile"));
 | 
				
			||||||
		assertHasMessageReader(readers, forClass(TestBean.class), null);
 | 
							assertHasMessageReader(readers, forClass(TestBean.class), null);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		WebBindingInitializer bindingInitializer = adapter.getWebBindingInitializer();
 | 
							WebBindingInitializer bindingInitializer = adapter.getWebBindingInitializer();
 | 
				
			||||||
| 
						 | 
					@ -189,7 +190,7 @@ public class WebFluxConfigurationSupportTests {
 | 
				
			||||||
		assertEquals(0, handler.getOrder());
 | 
							assertEquals(0, handler.getOrder());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		List<HttpMessageWriter<?>> writers = handler.getMessageWriters();
 | 
							List<HttpMessageWriter<?>> writers = handler.getMessageWriters();
 | 
				
			||||||
		assertEquals(9, writers.size());
 | 
							assertEquals(10, writers.size());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		assertHasMessageWriter(writers, forClass(byte[].class), APPLICATION_OCTET_STREAM);
 | 
							assertHasMessageWriter(writers, forClass(byte[].class), APPLICATION_OCTET_STREAM);
 | 
				
			||||||
		assertHasMessageWriter(writers, forClass(ByteBuffer.class), APPLICATION_OCTET_STREAM);
 | 
							assertHasMessageWriter(writers, forClass(ByteBuffer.class), APPLICATION_OCTET_STREAM);
 | 
				
			||||||
| 
						 | 
					@ -197,6 +198,7 @@ public class WebFluxConfigurationSupportTests {
 | 
				
			||||||
		assertHasMessageWriter(writers, forClass(Resource.class), IMAGE_PNG);
 | 
							assertHasMessageWriter(writers, forClass(Resource.class), IMAGE_PNG);
 | 
				
			||||||
		assertHasMessageWriter(writers, forClass(TestBean.class), APPLICATION_XML);
 | 
							assertHasMessageWriter(writers, forClass(TestBean.class), APPLICATION_XML);
 | 
				
			||||||
		assertHasMessageWriter(writers, forClass(TestBean.class), APPLICATION_JSON);
 | 
							assertHasMessageWriter(writers, forClass(TestBean.class), APPLICATION_JSON);
 | 
				
			||||||
 | 
							assertHasMessageWriter(writers, forClass(TestBean.class), new MediaType("application", "x-jackson-smile"));
 | 
				
			||||||
		assertHasMessageWriter(writers, forClass(TestBean.class), MediaType.parseMediaType("text/event-stream"));
 | 
							assertHasMessageWriter(writers, forClass(TestBean.class), MediaType.parseMediaType("text/event-stream"));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		name = "webFluxContentTypeResolver";
 | 
							name = "webFluxContentTypeResolver";
 | 
				
			||||||
| 
						 | 
					@ -215,7 +217,7 @@ public class WebFluxConfigurationSupportTests {
 | 
				
			||||||
		assertEquals(100, handler.getOrder());
 | 
							assertEquals(100, handler.getOrder());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		List<HttpMessageWriter<?>> writers = handler.getMessageWriters();
 | 
							List<HttpMessageWriter<?>> writers = handler.getMessageWriters();
 | 
				
			||||||
		assertEquals(9, writers.size());
 | 
							assertEquals(10, writers.size());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		assertHasMessageWriter(writers, forClass(byte[].class), APPLICATION_OCTET_STREAM);
 | 
							assertHasMessageWriter(writers, forClass(byte[].class), APPLICATION_OCTET_STREAM);
 | 
				
			||||||
		assertHasMessageWriter(writers, forClass(ByteBuffer.class), APPLICATION_OCTET_STREAM);
 | 
							assertHasMessageWriter(writers, forClass(ByteBuffer.class), APPLICATION_OCTET_STREAM);
 | 
				
			||||||
| 
						 | 
					@ -223,6 +225,7 @@ public class WebFluxConfigurationSupportTests {
 | 
				
			||||||
		assertHasMessageWriter(writers, forClass(Resource.class), IMAGE_PNG);
 | 
							assertHasMessageWriter(writers, forClass(Resource.class), IMAGE_PNG);
 | 
				
			||||||
		assertHasMessageWriter(writers, forClass(TestBean.class), APPLICATION_XML);
 | 
							assertHasMessageWriter(writers, forClass(TestBean.class), APPLICATION_XML);
 | 
				
			||||||
		assertHasMessageWriter(writers, forClass(TestBean.class), APPLICATION_JSON);
 | 
							assertHasMessageWriter(writers, forClass(TestBean.class), APPLICATION_JSON);
 | 
				
			||||||
 | 
							assertHasMessageWriter(writers, forClass(TestBean.class), new MediaType("application", "x-jackson-smile"));
 | 
				
			||||||
		assertHasMessageWriter(writers, forClass(TestBean.class), null);
 | 
							assertHasMessageWriter(writers, forClass(TestBean.class), null);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		name = "webFluxContentTypeResolver";
 | 
							name = "webFluxContentTypeResolver";
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in New Issue