25
25
import com .fasterxml .jackson .annotation .JsonTypeInfo ;
26
26
import com .fasterxml .jackson .annotation .JsonTypeInfo .As ;
27
27
import com .fasterxml .jackson .core .JsonGenerator ;
28
- import com .fasterxml .jackson .core .JsonProcessingException ;
29
28
import com .fasterxml .jackson .databind .ObjectMapper ;
30
29
import com .fasterxml .jackson .databind .ObjectMapper .DefaultTyping ;
31
30
import com .fasterxml .jackson .databind .SerializerProvider ;
37
36
38
37
/**
39
38
* 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}.
40
42
*
41
43
* @author Christoph Strobl
42
44
* @author Mark Paluch
@@ -47,6 +49,10 @@ public class GenericJackson2JsonRedisSerializer implements RedisSerializer<Objec
47
49
48
50
private final ObjectMapper mapper ;
49
51
52
+ private final JacksonObjectReader reader ;
53
+
54
+ private final JacksonObjectWriter writer ;
55
+
50
56
/**
51
57
* Creates {@link GenericJackson2JsonRedisSerializer} and configures {@link ObjectMapper} for default typing.
52
58
*/
@@ -59,13 +65,30 @@ public GenericJackson2JsonRedisSerializer() {
59
65
* given {@literal name}. In case of an {@literal empty} or {@literal null} String the default
60
66
* {@link JsonTypeInfo.Id#CLASS} will be used.
61
67
*
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}.
63
69
* @see ObjectMapper#activateDefaultTypingAsProperty(PolymorphicTypeValidator, DefaultTyping, String)
64
70
* @see ObjectMapper#activateDefaultTyping(PolymorphicTypeValidator, DefaultTyping, As)
65
71
*/
66
72
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 ) {
67
90
68
- this (new ObjectMapper ());
91
+ this (new ObjectMapper (), reader , writer );
69
92
70
93
// simply setting {@code mapper.disable(SerializationFeature.FAIL_ON_EMPTY_BEANS)} does not help here since we need
71
94
// the type hint embedded for deserialization using the default typing feature.
@@ -87,9 +110,29 @@ public GenericJackson2JsonRedisSerializer(@Nullable String classPropertyTypeName
87
110
* @param mapper must not be {@literal null}.
88
111
*/
89
112
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 ) {
90
128
91
129
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
+
92
133
this .mapper = mapper ;
134
+ this .reader = reader ;
135
+ this .writer = writer ;
93
136
}
94
137
95
138
/**
@@ -116,8 +159,8 @@ public byte[] serialize(@Nullable Object source) throws SerializationException {
116
159
}
117
160
118
161
try {
119
- return mapper . writeValueAsBytes ( source );
120
- } catch (JsonProcessingException e ) {
162
+ return writer . write ( mapper , source );
163
+ } catch (IOException e ) {
121
164
throw new SerializationException ("Could not write JSON: " + e .getMessage (), e );
122
165
}
123
166
}
@@ -134,6 +177,7 @@ public Object deserialize(@Nullable byte[] source) throws SerializationException
134
177
* @throws SerializationException
135
178
*/
136
179
@ Nullable
180
+ @ SuppressWarnings ("unchecked" )
137
181
public <T > T deserialize (@ Nullable byte [] source , Class <T > type ) throws SerializationException {
138
182
139
183
Assert .notNull (type ,
@@ -144,7 +188,7 @@ public <T> T deserialize(@Nullable byte[] source, Class<T> type) throws Serializ
144
188
}
145
189
146
190
try {
147
- return mapper . readValue ( source , type );
191
+ return ( T ) reader . read ( mapper , source , mapper . getTypeFactory (). constructType ( type ) );
148
192
} catch (Exception ex ) {
149
193
throw new SerializationException ("Could not read JSON: " + ex .getMessage (), ex );
150
194
}
@@ -172,8 +216,7 @@ private static class NullValueSerializer extends StdSerializer<NullValue> {
172
216
}
173
217
174
218
@ 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 {
177
220
178
221
jgen .writeStartObject ();
179
222
jgen .writeStringField (classIdentifier , NullValue .class .getName ());
@@ -186,4 +229,5 @@ public void serializeWithType(NullValue value, JsonGenerator gen, SerializerProv
186
229
serialize (value , gen , serializers );
187
230
}
188
231
}
232
+
189
233
}
0 commit comments