Skip to content

Commit eb7dfbc

Browse files
mp911dechristophstrobl
authored andcommitted
Introduce JacksonObjectReader and JacksonObjectWriter function interfaces to customize JSON (de)serialization.
We now encapsulate serialization and deserialization operations as JacksonObjectWriter and JacksonObjectReader functions to allow customization of Jackson serialization. Closes: #2322 Original Pull Request: #2332
1 parent 862decf commit eb7dfbc

File tree

6 files changed

+239
-11
lines changed

6 files changed

+239
-11
lines changed

src/main/java/org/springframework/data/redis/serializer/GenericJackson2JsonRedisSerializer.java

+52-8
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,6 @@
2525
import com.fasterxml.jackson.annotation.JsonTypeInfo;
2626
import com.fasterxml.jackson.annotation.JsonTypeInfo.As;
2727
import com.fasterxml.jackson.core.JsonGenerator;
28-
import com.fasterxml.jackson.core.JsonProcessingException;
2928
import com.fasterxml.jackson.databind.ObjectMapper;
3029
import com.fasterxml.jackson.databind.ObjectMapper.DefaultTyping;
3130
import com.fasterxml.jackson.databind.SerializerProvider;
@@ -37,6 +36,9 @@
3736

3837
/**
3938
* Generic Jackson 2-based {@link RedisSerializer} that maps {@link Object objects} to JSON using dynamic typing.
39+
* <p>
40+
* JSON reading and writing can be customized by configuring {@link JacksonObjectReader} respective
41+
* {@link JacksonObjectWriter}.
4042
*
4143
* @author Christoph Strobl
4244
* @author Mark Paluch
@@ -47,6 +49,10 @@ public class GenericJackson2JsonRedisSerializer implements RedisSerializer<Objec
4749

4850
private final ObjectMapper mapper;
4951

52+
private final JacksonObjectReader reader;
53+
54+
private final JacksonObjectWriter writer;
55+
5056
/**
5157
* Creates {@link GenericJackson2JsonRedisSerializer} and configures {@link ObjectMapper} for default typing.
5258
*/
@@ -59,13 +65,30 @@ public GenericJackson2JsonRedisSerializer() {
5965
* given {@literal name}. In case of an {@literal empty} or {@literal null} String the default
6066
* {@link JsonTypeInfo.Id#CLASS} will be used.
6167
*
62-
* @param classPropertyTypeName Name of the JSON property holding type information. Can be {@literal null}.
68+
* @param classPropertyTypeName name of the JSON property holding type information. Can be {@literal null}.
6369
* @see ObjectMapper#activateDefaultTypingAsProperty(PolymorphicTypeValidator, DefaultTyping, String)
6470
* @see ObjectMapper#activateDefaultTyping(PolymorphicTypeValidator, DefaultTyping, As)
6571
*/
6672
public GenericJackson2JsonRedisSerializer(@Nullable String classPropertyTypeName) {
73+
this(classPropertyTypeName, JacksonObjectReader.create(), JacksonObjectWriter.create());
74+
}
75+
76+
/**
77+
* Creates {@link GenericJackson2JsonRedisSerializer} and configures {@link ObjectMapper} for default typing using the
78+
* given {@literal name}. In case of an {@literal empty} or {@literal null} String the default
79+
* {@link JsonTypeInfo.Id#CLASS} will be used.
80+
*
81+
* @param classPropertyTypeName name of the JSON property holding type information. Can be {@literal null}.
82+
* @param reader the {@link JacksonObjectReader} function to read objects using {@link ObjectMapper}.
83+
* @param writer the {@link JacksonObjectWriter} function to write objects using {@link ObjectMapper}.
84+
* @see ObjectMapper#activateDefaultTypingAsProperty(PolymorphicTypeValidator, DefaultTyping, String)
85+
* @see ObjectMapper#activateDefaultTyping(PolymorphicTypeValidator, DefaultTyping, As)
86+
* @since 3.0
87+
*/
88+
public GenericJackson2JsonRedisSerializer(@Nullable String classPropertyTypeName, JacksonObjectReader reader,
89+
JacksonObjectWriter writer) {
6790

68-
this(new ObjectMapper());
91+
this(new ObjectMapper(), reader, writer);
6992

7093
// simply setting {@code mapper.disable(SerializationFeature.FAIL_ON_EMPTY_BEANS)} does not help here since we need
7194
// the type hint embedded for deserialization using the default typing feature.
@@ -87,9 +110,29 @@ public GenericJackson2JsonRedisSerializer(@Nullable String classPropertyTypeName
87110
* @param mapper must not be {@literal null}.
88111
*/
89112
public GenericJackson2JsonRedisSerializer(ObjectMapper mapper) {
113+
this(mapper, JacksonObjectReader.create(), JacksonObjectWriter.create());
114+
}
115+
116+
/**
117+
* Setting a custom-configured {@link ObjectMapper} is one way to take further control of the JSON serialization
118+
* process. For example, an extended {@link SerializerFactory} can be configured that provides custom serializers for
119+
* specific types.
120+
*
121+
* @param mapper must not be {@literal null}.
122+
* @param reader the {@link JacksonObjectReader} function to read objects using {@link ObjectMapper}.
123+
* @param writer the {@link JacksonObjectWriter} function to write objects using {@link ObjectMapper}.
124+
* @since 3.0
125+
*/
126+
public GenericJackson2JsonRedisSerializer(ObjectMapper mapper, JacksonObjectReader reader,
127+
JacksonObjectWriter writer) {
90128

91129
Assert.notNull(mapper, "ObjectMapper must not be null");
130+
Assert.notNull(reader, "Reader must not be null");
131+
Assert.notNull(writer, "Writer must not be null");
132+
92133
this.mapper = mapper;
134+
this.reader = reader;
135+
this.writer = writer;
93136
}
94137

95138
/**
@@ -116,8 +159,8 @@ public byte[] serialize(@Nullable Object source) throws SerializationException {
116159
}
117160

118161
try {
119-
return mapper.writeValueAsBytes(source);
120-
} catch (JsonProcessingException e) {
162+
return writer.write(mapper, source);
163+
} catch (IOException e) {
121164
throw new SerializationException("Could not write JSON: " + e.getMessage(), e);
122165
}
123166
}
@@ -134,6 +177,7 @@ public Object deserialize(@Nullable byte[] source) throws SerializationException
134177
* @throws SerializationException
135178
*/
136179
@Nullable
180+
@SuppressWarnings("unchecked")
137181
public <T> T deserialize(@Nullable byte[] source, Class<T> type) throws SerializationException {
138182

139183
Assert.notNull(type,
@@ -144,7 +188,7 @@ public <T> T deserialize(@Nullable byte[] source, Class<T> type) throws Serializ
144188
}
145189

146190
try {
147-
return mapper.readValue(source, type);
191+
return (T) reader.read(mapper, source, mapper.getTypeFactory().constructType(type));
148192
} catch (Exception ex) {
149193
throw new SerializationException("Could not read JSON: " + ex.getMessage(), ex);
150194
}
@@ -172,8 +216,7 @@ private static class NullValueSerializer extends StdSerializer<NullValue> {
172216
}
173217

174218
@Override
175-
public void serialize(NullValue value, JsonGenerator jgen, SerializerProvider provider)
176-
throws IOException {
219+
public void serialize(NullValue value, JsonGenerator jgen, SerializerProvider provider) throws IOException {
177220

178221
jgen.writeStartObject();
179222
jgen.writeStringField(classIdentifier, NullValue.class.getName());
@@ -186,4 +229,5 @@ public void serializeWithType(NullValue value, JsonGenerator gen, SerializerProv
186229
serialize(value, gen, serializers);
187230
}
188231
}
232+
189233
}

src/main/java/org/springframework/data/redis/serializer/Jackson2JsonRedisSerializer.java

+31-3
Original file line numberDiff line numberDiff line change
@@ -31,10 +31,14 @@
3131
* <a href="https://github.com/FasterXML/jackson-core">Jackson's</a> and
3232
* <a href="https://github.com/FasterXML/jackson-databind">Jackson Databind</a> {@link ObjectMapper}.
3333
* <p>
34-
* This converter can be used to bind to typed beans, or untyped {@link java.util.HashMap HashMap} instances.
34+
* This serializer can be used to bind to typed beans, or untyped {@link java.util.HashMap HashMap} instances.
3535
* <b>Note:</b>Null objects are serialized as empty arrays and vice versa.
36+
* <p>
37+
* JSON reading and writing can be customized by configuring {@link JacksonObjectReader} respective
38+
* {@link JacksonObjectWriter}.
3639
*
3740
* @author Thomas Darimont
41+
* @author Mark Paluch
3842
* @since 1.2
3943
*/
4044
public class Jackson2JsonRedisSerializer<T> implements RedisSerializer<T> {
@@ -45,6 +49,10 @@ public class Jackson2JsonRedisSerializer<T> implements RedisSerializer<T> {
4549

4650
private ObjectMapper objectMapper = new ObjectMapper();
4751

52+
private JacksonObjectReader reader = JacksonObjectReader.create();
53+
54+
private JacksonObjectWriter writer = JacksonObjectWriter.create();
55+
4856
/**
4957
* Creates a new {@link Jackson2JsonRedisSerializer} for the given target {@link Class}.
5058
*
@@ -70,7 +78,7 @@ public T deserialize(@Nullable byte[] bytes) throws SerializationException {
7078
return null;
7179
}
7280
try {
73-
return (T) this.objectMapper.readValue(bytes, 0, bytes.length, javaType);
81+
return (T) this.reader.read(this.objectMapper, bytes, javaType);
7482
} catch (Exception ex) {
7583
throw new SerializationException("Could not read JSON: " + ex.getMessage(), ex);
7684
}
@@ -83,7 +91,7 @@ public byte[] serialize(@Nullable Object t) throws SerializationException {
8391
return SerializationUtils.EMPTY_ARRAY;
8492
}
8593
try {
86-
return this.objectMapper.writeValueAsBytes(t);
94+
return this.writer.write(this.objectMapper, t);
8795
} catch (Exception ex) {
8896
throw new SerializationException("Could not write JSON: " + ex.getMessage(), ex);
8997
}
@@ -104,6 +112,26 @@ public void setObjectMapper(ObjectMapper objectMapper) {
104112
this.objectMapper = objectMapper;
105113
}
106114

115+
/**
116+
* Sets the {@link JacksonObjectReader} for this serializer. Setting the reader allows customization of the JSON
117+
* deserialization.
118+
*
119+
* @since 3.0
120+
*/
121+
public void setReader(JacksonObjectReader reader) {
122+
this.reader = reader;
123+
}
124+
125+
/**
126+
* Sets the {@link JacksonObjectWriter} for this serializer. Setting the reader allows customization of the JSON
127+
* serialization.
128+
*
129+
* @since 3.0
130+
*/
131+
public void setWriter(JacksonObjectWriter writer) {
132+
this.writer = writer;
133+
}
134+
107135
/**
108136
* Returns the Jackson {@link JavaType} for the specific class.
109137
* <p>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,57 @@
1+
/*
2+
* Copyright 2022 the original author or authors.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package org.springframework.data.redis.serializer;
17+
18+
import java.io.IOException;
19+
import java.io.InputStream;
20+
21+
import com.fasterxml.jackson.databind.JavaType;
22+
import com.fasterxml.jackson.databind.ObjectMapper;
23+
24+
/**
25+
* Defines the contract for Object Mapping readers. Implementations of this interface can deserialize a given byte array
26+
* holding JSON to an Object considering the target type.
27+
* <p>
28+
* Reader functions can customize how the actual JSON is being deserialized by e.g. obtaining a customized
29+
* {@link com.fasterxml.jackson.databind.ObjectReader} applying serialization features, date formats, or views.
30+
*
31+
* @author Mark Paluch
32+
* @since 3.0
33+
*/
34+
@FunctionalInterface
35+
public interface JacksonObjectReader {
36+
37+
/**
38+
* Read an object graph from the given root JSON into a Java object considering the {@link JavaType}.
39+
*
40+
* @param mapper the object mapper to use.
41+
* @param source the JSON to deserialize.
42+
* @param type the Java target type
43+
* @return the deserialized Java object.
44+
* @throws IOException if an I/O error or JSON deserialization error occurs.
45+
*/
46+
Object read(ObjectMapper mapper, byte[] source, JavaType type) throws IOException;
47+
48+
/**
49+
* Create a default {@link JacksonObjectReader} delegating to {@link ObjectMapper#readValue(InputStream, JavaType)}.
50+
*
51+
* @return the default {@link JacksonObjectReader}.
52+
*/
53+
static JacksonObjectReader create() {
54+
return (mapper, source, type) -> mapper.readValue(source, 0, source.length, type);
55+
}
56+
57+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,54 @@
1+
/*
2+
* Copyright 2022 the original author or authors.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package org.springframework.data.redis.serializer;
17+
18+
import java.io.IOException;
19+
20+
import com.fasterxml.jackson.databind.ObjectMapper;
21+
22+
/**
23+
* Defines the contract for Object Mapping writers. Implementations of this interface can serialize a given Object to a
24+
* {@code byte[]} containing JSON.
25+
* <p>
26+
* Writer functions can customize how the actual JSON is being written by e.g. obtaining a customized
27+
* {@link com.fasterxml.jackson.databind.ObjectWriter} applying serialization features, date formats, or views.
28+
*
29+
* @author Mark Paluch
30+
* @since 3.0
31+
*/
32+
@FunctionalInterface
33+
public interface JacksonObjectWriter {
34+
35+
/**
36+
* Write the object graph with the given root {@code source} as byte array.
37+
*
38+
* @param mapper the object mapper to use.
39+
* @param source the root of the object graph to marshal.
40+
* @return a byte array containing the serialized object graph.
41+
* @throws IOException if an I/O error or JSON serialization error occurs.
42+
*/
43+
byte[] write(ObjectMapper mapper, Object source) throws IOException;
44+
45+
/**
46+
* Create a default {@link JacksonObjectWriter} delegating to {@link ObjectMapper#writeValueAsBytes(Object)}.
47+
*
48+
* @return the default {@link JacksonObjectWriter}.
49+
*/
50+
static JacksonObjectWriter create() {
51+
return ObjectMapper::writeValueAsBytes;
52+
}
53+
54+
}

src/test/java/org/springframework/data/redis/serializer/GenericJackson2JsonRedisSerializerUnitTests.java

+34
Original file line numberDiff line numberDiff line change
@@ -26,11 +26,13 @@
2626

2727
import org.junit.jupiter.api.Test;
2828
import org.mockito.Mockito;
29+
2930
import org.springframework.beans.BeanUtils;
3031
import org.springframework.cache.support.NullValue;
3132

3233
import com.fasterxml.jackson.annotation.JsonTypeInfo;
3334
import com.fasterxml.jackson.annotation.JsonTypeInfo.As;
35+
import com.fasterxml.jackson.annotation.JsonView;
3436
import com.fasterxml.jackson.core.JsonGenerationException;
3537
import com.fasterxml.jackson.core.JsonProcessingException;
3638
import com.fasterxml.jackson.databind.JsonMappingException;
@@ -162,6 +164,24 @@ void deserializeShouldBeAbleToRestoreFinalObjectAfterSerialization() {
162164
assertThat(serializer.deserialize(serializer.serialize(source))).isEqualTo(source);
163165
}
164166

167+
@Test // GH-2322
168+
void shouldConsiderWriter() {
169+
170+
User user = new User();
171+
user.email = "[email protected]";
172+
user.id = 42;
173+
user.name = "Walter White";
174+
175+
GenericJackson2JsonRedisSerializer serializer = new GenericJackson2JsonRedisSerializer((String) null,
176+
JacksonObjectReader.create(), (mapper, source) -> {
177+
return mapper.writerWithView(Views.Basic.class).writeValueAsBytes(source);
178+
});
179+
180+
byte[] result = serializer.serialize(user);
181+
182+
assertThat(new String(result)).contains("id").contains("name").doesNotContain("email");
183+
}
184+
165185
private static void serializeAndDeserializeNullValue(GenericJackson2JsonRedisSerializer serializer) {
166186

167187
NullValue nv = BeanUtils.instantiateClass(NullValue.class);
@@ -252,4 +272,18 @@ public boolean equals(Object obj) {
252272
}
253273
}
254274

275+
public class User {
276+
@JsonView(Views.Basic.class) public int id;
277+
@JsonView(Views.Basic.class) public String name;
278+
@JsonView(Views.Detailed.class) public String email;
279+
@JsonView(Views.Detailed.class) public String mobile;
280+
}
281+
282+
public class Views {
283+
interface Basic {}
284+
285+
interface Detailed {}
286+
287+
}
288+
255289
}

0 commit comments

Comments
 (0)