/*
 * Copyright 2002-2018 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.converter.json;

import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.Arrays;
import java.util.LinkedHashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.TimeZone;

import com.fasterxml.jackson.annotation.JsonAutoDetect;
import com.fasterxml.jackson.annotation.JsonFilter;
import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.annotation.PropertyAccessor;
import com.fasterxml.jackson.core.JsonFactory;
import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.databind.AnnotationIntrospector;
import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.JsonDeserializer;
import com.fasterxml.jackson.databind.JsonSerializer;
import com.fasterxml.jackson.databind.KeyDeserializer;
import com.fasterxml.jackson.databind.MapperFeature;
import com.fasterxml.jackson.databind.Module;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.PropertyNamingStrategy;
import com.fasterxml.jackson.databind.SerializationFeature;
import com.fasterxml.jackson.databind.cfg.HandlerInstantiator;
import com.fasterxml.jackson.databind.jsontype.TypeResolverBuilder;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.ser.FilterProvider;
import com.fasterxml.jackson.dataformat.cbor.CBORFactory;
import com.fasterxml.jackson.dataformat.smile.SmileFactory;
import com.fasterxml.jackson.dataformat.xml.JacksonXmlModule;
import com.fasterxml.jackson.dataformat.xml.XmlFactory;
import com.fasterxml.jackson.dataformat.xml.XmlMapper;
import org.apache.commons.logging.Log;

import org.springframework.beans.BeanUtils;
import org.springframework.beans.FatalBeanException;
import org.springframework.context.ApplicationContext;
import org.springframework.core.KotlinDetector;
import org.springframework.http.HttpLogging;
import org.springframework.lang.Nullable;
import org.springframework.util.Assert;
import org.springframework.util.ClassUtils;
import org.springframework.util.StringUtils;
import org.springframework.util.xml.StaxUtils;

A builder used to create ObjectMapper instances with a fluent API.

It customizes Jackson's default properties with the following ones:

It also automatically registers the following well-known modules if they are detected on the classpath:

Compatible with Jackson 2.6 and higher, as of Spring 4.3.

Author:Sebastien Deleuze, Juergen Hoeller, Tadaya Tsuyukubo, Eddú Meléndez
See Also:
Since:4.1.1
/** * A builder used to create {@link ObjectMapper} instances with a fluent API. * * <p>It customizes Jackson's default properties with the following ones: * <ul> * <li>{@link MapperFeature#DEFAULT_VIEW_INCLUSION} is disabled</li> * <li>{@link DeserializationFeature#FAIL_ON_UNKNOWN_PROPERTIES} is disabled</li> * </ul> * * <p>It also automatically registers the following well-known modules if they are * detected on the classpath: * <ul> * <li><a href="https://github.com/FasterXML/jackson-datatype-jdk8">jackson-datatype-jdk8</a>: * support for other Java 8 types like {@link java.util.Optional}</li> * <li><a href="https://github.com/FasterXML/jackson-datatype-jsr310">jackson-datatype-jsr310</a>: * support for Java 8 Date & Time API types</li> * <li><a href="https://github.com/FasterXML/jackson-datatype-joda">jackson-datatype-joda</a>: * support for Joda-Time types</li> * <li><a href="https://github.com/FasterXML/jackson-module-kotlin">jackson-module-kotlin</a>: * support for Kotlin classes and data classes</li> * </ul> * * <p>Compatible with Jackson 2.6 and higher, as of Spring 4.3. * * @author Sebastien Deleuze * @author Juergen Hoeller * @author Tadaya Tsuyukubo * @author Eddú Meléndez * @since 4.1.1 * @see #build() * @see #configure(ObjectMapper) * @see Jackson2ObjectMapperFactoryBean */
public class Jackson2ObjectMapperBuilder { private static volatile boolean kotlinWarningLogged = false; private final Log logger = HttpLogging.forLogName(getClass()); private final Map<Class<?>, Class<?>> mixIns = new LinkedHashMap<>(); private final Map<Class<?>, JsonSerializer<?>> serializers = new LinkedHashMap<>(); private final Map<Class<?>, JsonDeserializer<?>> deserializers = new LinkedHashMap<>(); private final Map<PropertyAccessor, JsonAutoDetect.Visibility> visibilities = new LinkedHashMap<>(); private final Map<Object, Boolean> features = new LinkedHashMap<>(); private boolean createXmlMapper = false; @Nullable private JsonFactory factory; @Nullable private DateFormat dateFormat; @Nullable private Locale locale; @Nullable private TimeZone timeZone; @Nullable private AnnotationIntrospector annotationIntrospector; @Nullable private PropertyNamingStrategy propertyNamingStrategy; @Nullable private TypeResolverBuilder<?> defaultTyping; @Nullable private JsonInclude.Include serializationInclusion; @Nullable private FilterProvider filters; @Nullable private List<Module> modules; @Nullable private Class<? extends Module>[] moduleClasses; private boolean findModulesViaServiceLoader = false; private boolean findWellKnownModules = true; private ClassLoader moduleClassLoader = getClass().getClassLoader(); @Nullable private HandlerInstantiator handlerInstantiator; @Nullable private ApplicationContext applicationContext; @Nullable private Boolean defaultUseWrapper;
If set to true, an XmlMapper will be created using its default constructor. This is only applicable to build() calls, not to configure calls.
/** * If set to {@code true}, an {@link XmlMapper} will be created using its * default constructor. This is only applicable to {@link #build()} calls, * not to {@link #configure} calls. */
public Jackson2ObjectMapperBuilder createXmlMapper(boolean createXmlMapper) { this.createXmlMapper = createXmlMapper; return this; }
Define the JsonFactory to be used to create the ObjectMapper instance.
Since:5.0
/** * Define the {@link JsonFactory} to be used to create the {@link ObjectMapper} * instance. * @since 5.0 */
public Jackson2ObjectMapperBuilder factory(JsonFactory factory) { this.factory = factory; return this; }
Define the format for date/time with the given DateFormat.

Note: Setting this property makes the exposed ObjectMapper non-thread-safe, according to Jackson's thread safety rules.

See Also:
/** * Define the format for date/time with the given {@link DateFormat}. * <p>Note: Setting this property makes the exposed {@link ObjectMapper} * non-thread-safe, according to Jackson's thread safety rules. * @see #simpleDateFormat(String) */
public Jackson2ObjectMapperBuilder dateFormat(DateFormat dateFormat) { this.dateFormat = dateFormat; return this; }
Define the date/time format with a SimpleDateFormat.

Note: Setting this property makes the exposed ObjectMapper non-thread-safe, according to Jackson's thread safety rules.

See Also:
/** * Define the date/time format with a {@link SimpleDateFormat}. * <p>Note: Setting this property makes the exposed {@link ObjectMapper} * non-thread-safe, according to Jackson's thread safety rules. * @see #dateFormat(DateFormat) */
public Jackson2ObjectMapperBuilder simpleDateFormat(String format) { this.dateFormat = new SimpleDateFormat(format); return this; }
Override the default Locale to use for formatting. Default value used is Locale.getDefault().
Since:4.1.5
/** * Override the default {@link Locale} to use for formatting. * Default value used is {@link Locale#getDefault()}. * @since 4.1.5 */
public Jackson2ObjectMapperBuilder locale(Locale locale) { this.locale = locale; return this; }
Override the default Locale to use for formatting. Default value used is Locale.getDefault().
Params:
  • localeString – the locale ID as a String representation
Since:4.1.5
/** * Override the default {@link Locale} to use for formatting. * Default value used is {@link Locale#getDefault()}. * @param localeString the locale ID as a String representation * @since 4.1.5 */
public Jackson2ObjectMapperBuilder locale(String localeString) { this.locale = StringUtils.parseLocale(localeString); return this; }
Override the default TimeZone to use for formatting. Default value used is UTC (NOT local timezone).
Since:4.1.5
/** * Override the default {@link TimeZone} to use for formatting. * Default value used is UTC (NOT local timezone). * @since 4.1.5 */
public Jackson2ObjectMapperBuilder timeZone(TimeZone timeZone) { this.timeZone = timeZone; return this; }
Override the default TimeZone to use for formatting. Default value used is UTC (NOT local timezone).
Params:
  • timeZoneString – the zone ID as a String representation
Since:4.1.5
/** * Override the default {@link TimeZone} to use for formatting. * Default value used is UTC (NOT local timezone). * @param timeZoneString the zone ID as a String representation * @since 4.1.5 */
public Jackson2ObjectMapperBuilder timeZone(String timeZoneString) { this.timeZone = StringUtils.parseTimeZoneString(timeZoneString); return this; }
Set an AnnotationIntrospector for both serialization and deserialization.
/** * Set an {@link AnnotationIntrospector} for both serialization and deserialization. */
public Jackson2ObjectMapperBuilder annotationIntrospector(AnnotationIntrospector annotationIntrospector) { this.annotationIntrospector = annotationIntrospector; return this; }
Specify a PropertyNamingStrategy to configure the ObjectMapper with.
/** * Specify a {@link com.fasterxml.jackson.databind.PropertyNamingStrategy} to * configure the {@link ObjectMapper} with. */
public Jackson2ObjectMapperBuilder propertyNamingStrategy(PropertyNamingStrategy propertyNamingStrategy) { this.propertyNamingStrategy = propertyNamingStrategy; return this; }
Specify a TypeResolverBuilder to use for Jackson's default typing.
Since:4.2.2
/** * Specify a {@link TypeResolverBuilder} to use for Jackson's default typing. * @since 4.2.2 */
public Jackson2ObjectMapperBuilder defaultTyping(TypeResolverBuilder<?> typeResolverBuilder) { this.defaultTyping = typeResolverBuilder; return this; }
Set a custom inclusion strategy for serialization.
See Also:
  • Include
/** * Set a custom inclusion strategy for serialization. * @see com.fasterxml.jackson.annotation.JsonInclude.Include */
public Jackson2ObjectMapperBuilder serializationInclusion(JsonInclude.Include serializationInclusion) { this.serializationInclusion = serializationInclusion; return this; }
Set the global filters to use in order to support @JsonFilter annotated POJO.
See Also:
Since:4.2
/** * Set the global filters to use in order to support {@link JsonFilter @JsonFilter} annotated POJO. * @since 4.2 * @see MappingJacksonValue#setFilters(FilterProvider) */
public Jackson2ObjectMapperBuilder filters(FilterProvider filters) { this.filters = filters; return this; }
Add mix-in annotations to use for augmenting specified class or interface.
Params:
  • target – class (or interface) whose annotations to effectively override
  • mixinSource – class (or interface) whose annotations are to be "added" to target's annotations as value
See Also:
Since:4.1.2
/** * Add mix-in annotations to use for augmenting specified class or interface. * @param target class (or interface) whose annotations to effectively override * @param mixinSource class (or interface) whose annotations are to be "added" * to target's annotations as value * @since 4.1.2 * @see com.fasterxml.jackson.databind.ObjectMapper#addMixInAnnotations(Class, Class) */
public Jackson2ObjectMapperBuilder mixIn(Class<?> target, Class<?> mixinSource) { this.mixIns.put(target, mixinSource); return this; }
Add mix-in annotations to use for augmenting specified class or interface.
Params:
  • mixIns – a Map of entries with target classes (or interface) whose annotations to effectively override as key and mix-in classes (or interface) whose annotations are to be "added" to target's annotations as value.
See Also:
Since:4.1.2
/** * Add mix-in annotations to use for augmenting specified class or interface. * @param mixIns a Map of entries with target classes (or interface) whose annotations * to effectively override as key and mix-in classes (or interface) whose * annotations are to be "added" to target's annotations as value. * @since 4.1.2 * @see com.fasterxml.jackson.databind.ObjectMapper#addMixInAnnotations(Class, Class) */
public Jackson2ObjectMapperBuilder mixIns(Map<Class<?>, Class<?>> mixIns) { this.mixIns.putAll(mixIns); return this; }
Configure custom serializers. Each serializer is registered for the type returned by JsonSerializer.handledType(), which must not be null.
See Also:
/** * Configure custom serializers. Each serializer is registered for the type * returned by {@link JsonSerializer#handledType()}, which must not be {@code null}. * @see #serializersByType(Map) */
public Jackson2ObjectMapperBuilder serializers(JsonSerializer<?>... serializers) { for (JsonSerializer<?> serializer : serializers) { Class<?> handledType = serializer.handledType(); if (handledType == null || handledType == Object.class) { throw new IllegalArgumentException("Unknown handled type in " + serializer.getClass().getName()); } this.serializers.put(serializer.handledType(), serializer); } return this; }
Configure a custom serializer for the given type.
See Also:
Since:4.1.2
/** * Configure a custom serializer for the given type. * @since 4.1.2 * @see #serializers(JsonSerializer...) */
public Jackson2ObjectMapperBuilder serializerByType(Class<?> type, JsonSerializer<?> serializer) { this.serializers.put(type, serializer); return this; }
Configure custom serializers for the given types.
See Also:
  • serializers(JsonSerializer...)
/** * Configure custom serializers for the given types. * @see #serializers(JsonSerializer...) */
public Jackson2ObjectMapperBuilder serializersByType(Map<Class<?>, JsonSerializer<?>> serializers) { this.serializers.putAll(serializers); return this; }
Configure custom deserializers. Each deserializer is registered for the type returned by JsonDeserializer.handledType(), which must not be null.
See Also:
Since:4.3
/** * Configure custom deserializers. Each deserializer is registered for the type * returned by {@link JsonDeserializer#handledType()}, which must not be {@code null}. * @since 4.3 * @see #deserializersByType(Map) */
public Jackson2ObjectMapperBuilder deserializers(JsonDeserializer<?>... deserializers) { for (JsonDeserializer<?> deserializer : deserializers) { Class<?> handledType = deserializer.handledType(); if (handledType == null || handledType == Object.class) { throw new IllegalArgumentException("Unknown handled type in " + deserializer.getClass().getName()); } this.deserializers.put(deserializer.handledType(), deserializer); } return this; }
Configure a custom deserializer for the given type.
Since:4.1.2
/** * Configure a custom deserializer for the given type. * @since 4.1.2 */
public Jackson2ObjectMapperBuilder deserializerByType(Class<?> type, JsonDeserializer<?> deserializer) { this.deserializers.put(type, deserializer); return this; }
Configure custom deserializers for the given types.
/** * Configure custom deserializers for the given types. */
public Jackson2ObjectMapperBuilder deserializersByType(Map<Class<?>, JsonDeserializer<?>> deserializers) { this.deserializers.putAll(deserializers); return this; }
Shortcut for MapperFeature.AUTO_DETECT_FIELDS option.
/** * Shortcut for {@link MapperFeature#AUTO_DETECT_FIELDS} option. */
public Jackson2ObjectMapperBuilder autoDetectFields(boolean autoDetectFields) { this.features.put(MapperFeature.AUTO_DETECT_FIELDS, autoDetectFields); return this; } /** * Shortcut for {@link MapperFeature#AUTO_DETECT_SETTERS}/ * {@link MapperFeature#AUTO_DETECT_GETTERS}/{@link MapperFeature#AUTO_DETECT_IS_GETTERS} * options. */ public Jackson2ObjectMapperBuilder autoDetectGettersSetters(boolean autoDetectGettersSetters) { this.features.put(MapperFeature.AUTO_DETECT_GETTERS, autoDetectGettersSetters); this.features.put(MapperFeature.AUTO_DETECT_SETTERS, autoDetectGettersSetters); this.features.put(MapperFeature.AUTO_DETECT_IS_GETTERS, autoDetectGettersSetters); return this; } /** * Shortcut for {@link MapperFeature#DEFAULT_VIEW_INCLUSION} option. */ public Jackson2ObjectMapperBuilder defaultViewInclusion(boolean defaultViewInclusion) { this.features.put(MapperFeature.DEFAULT_VIEW_INCLUSION, defaultViewInclusion); return this; } /** * Shortcut for {@link DeserializationFeature#FAIL_ON_UNKNOWN_PROPERTIES} option. */ public Jackson2ObjectMapperBuilder failOnUnknownProperties(boolean failOnUnknownProperties) { this.features.put(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, failOnUnknownProperties); return this; } /** * Shortcut for {@link SerializationFeature#FAIL_ON_EMPTY_BEANS} option. */ public Jackson2ObjectMapperBuilder failOnEmptyBeans(boolean failOnEmptyBeans) { this.features.put(SerializationFeature.FAIL_ON_EMPTY_BEANS, failOnEmptyBeans); return this; } /** * Shortcut for {@link SerializationFeature#INDENT_OUTPUT} option. */ public Jackson2ObjectMapperBuilder indentOutput(boolean indentOutput) { this.features.put(SerializationFeature.INDENT_OUTPUT, indentOutput); return this; }
Define if a wrapper will be used for indexed (List, array) properties or not by default (only applies to XmlMapper).
Since:4.3
/** * Define if a wrapper will be used for indexed (List, array) properties or not by * default (only applies to {@link XmlMapper}). * @since 4.3 */
public Jackson2ObjectMapperBuilder defaultUseWrapper(boolean defaultUseWrapper) { this.defaultUseWrapper = defaultUseWrapper; return this; }
Specify visibility to limit what kind of properties are auto-detected.
See Also:
Since:5.1
/** * Specify visibility to limit what kind of properties are auto-detected. * @since 5.1 * @see com.fasterxml.jackson.annotation.PropertyAccessor * @see com.fasterxml.jackson.annotation.JsonAutoDetect.Visibility */
public Jackson2ObjectMapperBuilder visibility(PropertyAccessor accessor, JsonAutoDetect.Visibility visibility) { this.visibilities.put(accessor, visibility); return this; }
Specify features to enable.
See Also:
/** * Specify features to enable. * @see com.fasterxml.jackson.core.JsonParser.Feature * @see com.fasterxml.jackson.core.JsonGenerator.Feature * @see com.fasterxml.jackson.databind.SerializationFeature * @see com.fasterxml.jackson.databind.DeserializationFeature * @see com.fasterxml.jackson.databind.MapperFeature */
public Jackson2ObjectMapperBuilder featuresToEnable(Object... featuresToEnable) { for (Object feature : featuresToEnable) { this.features.put(feature, Boolean.TRUE); } return this; }
Specify features to disable.
See Also:
/** * Specify features to disable. * @see com.fasterxml.jackson.core.JsonParser.Feature * @see com.fasterxml.jackson.core.JsonGenerator.Feature * @see com.fasterxml.jackson.databind.SerializationFeature * @see com.fasterxml.jackson.databind.DeserializationFeature * @see com.fasterxml.jackson.databind.MapperFeature */
public Jackson2ObjectMapperBuilder featuresToDisable(Object... featuresToDisable) { for (Object feature : featuresToDisable) { this.features.put(feature, Boolean.FALSE); } return this; }
Specify one or more modules to be registered with the ObjectMapper.

Note: If this is set, no finding of modules is going to happen - not by Jackson, and not by Spring either (see findModulesViaServiceLoader). As a consequence, specifying an empty list here will suppress any kind of module detection.

Specify either this or modulesToInstall, not both.

See Also:
Since:4.1.5
/** * Specify one or more modules to be registered with the {@link ObjectMapper}. * <p>Note: If this is set, no finding of modules is going to happen - not by * Jackson, and not by Spring either (see {@link #findModulesViaServiceLoader}). * As a consequence, specifying an empty list here will suppress any kind of * module detection. * <p>Specify either this or {@link #modulesToInstall}, not both. * @since 4.1.5 * @see #modules(List) * @see com.fasterxml.jackson.databind.Module */
public Jackson2ObjectMapperBuilder modules(Module... modules) { return modules(Arrays.asList(modules)); }
Set a complete list of modules to be registered with the ObjectMapper.

Note: If this is set, no finding of modules is going to happen - not by Jackson, and not by Spring either (see findModulesViaServiceLoader). As a consequence, specifying an empty list here will suppress any kind of module detection.

Specify either this or modulesToInstall, not both.

See Also:
/** * Set a complete list of modules to be registered with the {@link ObjectMapper}. * <p>Note: If this is set, no finding of modules is going to happen - not by * Jackson, and not by Spring either (see {@link #findModulesViaServiceLoader}). * As a consequence, specifying an empty list here will suppress any kind of * module detection. * <p>Specify either this or {@link #modulesToInstall}, not both. * @see #modules(Module...) * @see com.fasterxml.jackson.databind.Module */
public Jackson2ObjectMapperBuilder modules(List<Module> modules) { this.modules = new LinkedList<>(modules); this.findModulesViaServiceLoader = false; this.findWellKnownModules = false; return this; }
Specify one or more modules to be registered with the ObjectMapper.

Modules specified here will be registered after Spring's autodetection of JSR-310 and Joda-Time, or Jackson's finding of modules (see findModulesViaServiceLoader), allowing to eventually override their configuration.

Specify either this or modules, not both.

See Also:
Since:4.1.5
/** * Specify one or more modules to be registered with the {@link ObjectMapper}. * <p>Modules specified here will be registered after * Spring's autodetection of JSR-310 and Joda-Time, or Jackson's * finding of modules (see {@link #findModulesViaServiceLoader}), * allowing to eventually override their configuration. * <p>Specify either this or {@link #modules}, not both. * @since 4.1.5 * @see com.fasterxml.jackson.databind.Module */
public Jackson2ObjectMapperBuilder modulesToInstall(Module... modules) { this.modules = Arrays.asList(modules); this.findWellKnownModules = true; return this; }
Specify one or more modules by class to be registered with the ObjectMapper.

Modules specified here will be registered after Spring's autodetection of JSR-310 and Joda-Time, or Jackson's finding of modules (see findModulesViaServiceLoader), allowing to eventually override their configuration.

Specify either this or modules, not both.

See Also:
/** * Specify one or more modules by class to be registered with * the {@link ObjectMapper}. * <p>Modules specified here will be registered after * Spring's autodetection of JSR-310 and Joda-Time, or Jackson's * finding of modules (see {@link #findModulesViaServiceLoader}), * allowing to eventually override their configuration. * <p>Specify either this or {@link #modules}, not both. * @see #modulesToInstall(Module...) * @see com.fasterxml.jackson.databind.Module */
@SuppressWarnings("unchecked") public Jackson2ObjectMapperBuilder modulesToInstall(Class<? extends Module>... modules) { this.moduleClasses = modules; this.findWellKnownModules = true; return this; }
Set whether to let Jackson find available modules via the JDK ServiceLoader, based on META-INF metadata in the classpath.

If this mode is not set, Spring's Jackson2ObjectMapperBuilder itself will try to find the JSR-310 and Joda-Time support modules on the classpath - provided that Java 8 and Joda-Time themselves are available, respectively.

See Also:
  • findModules.findModules()
/** * Set whether to let Jackson find available modules via the JDK ServiceLoader, * based on META-INF metadata in the classpath. * <p>If this mode is not set, Spring's Jackson2ObjectMapperBuilder itself * will try to find the JSR-310 and Joda-Time support modules on the classpath - * provided that Java 8 and Joda-Time themselves are available, respectively. * @see com.fasterxml.jackson.databind.ObjectMapper#findModules() */
public Jackson2ObjectMapperBuilder findModulesViaServiceLoader(boolean findModules) { this.findModulesViaServiceLoader = findModules; return this; }
Set the ClassLoader to use for loading Jackson extension modules.
/** * Set the ClassLoader to use for loading Jackson extension modules. */
public Jackson2ObjectMapperBuilder moduleClassLoader(ClassLoader moduleClassLoader) { this.moduleClassLoader = moduleClassLoader; return this; }
Customize the construction of Jackson handlers (JsonSerializer, JsonDeserializer, KeyDeserializer, TypeResolverBuilder and TypeIdResolver).
See Also:
Since:4.1.3
/** * Customize the construction of Jackson handlers ({@link JsonSerializer}, {@link JsonDeserializer}, * {@link KeyDeserializer}, {@code TypeResolverBuilder} and {@code TypeIdResolver}). * @since 4.1.3 * @see Jackson2ObjectMapperBuilder#applicationContext(ApplicationContext) */
public Jackson2ObjectMapperBuilder handlerInstantiator(HandlerInstantiator handlerInstantiator) { this.handlerInstantiator = handlerInstantiator; return this; }
Set the Spring ApplicationContext in order to autowire Jackson handlers (JsonSerializer, JsonDeserializer, KeyDeserializer, TypeResolverBuilder and TypeIdResolver).
See Also:
Since:4.1.3
/** * Set the Spring {@link ApplicationContext} in order to autowire Jackson handlers ({@link JsonSerializer}, * {@link JsonDeserializer}, {@link KeyDeserializer}, {@code TypeResolverBuilder} and {@code TypeIdResolver}). * @since 4.1.3 * @see SpringHandlerInstantiator */
public Jackson2ObjectMapperBuilder applicationContext(ApplicationContext applicationContext) { this.applicationContext = applicationContext; return this; }
Build a new ObjectMapper instance.

Each build operation produces an independent ObjectMapper instance. The builder's settings can get modified, with a subsequent build operation then producing a new ObjectMapper based on the most recent settings.

Returns:the newly built ObjectMapper
/** * Build a new {@link ObjectMapper} instance. * <p>Each build operation produces an independent {@link ObjectMapper} instance. * The builder's settings can get modified, with a subsequent build operation * then producing a new {@link ObjectMapper} based on the most recent settings. * @return the newly built ObjectMapper */
@SuppressWarnings("unchecked") public <T extends ObjectMapper> T build() { ObjectMapper mapper; if (this.createXmlMapper) { mapper = (this.defaultUseWrapper != null ? new XmlObjectMapperInitializer().create(this.defaultUseWrapper) : new XmlObjectMapperInitializer().create()); } else { mapper = (this.factory != null ? new ObjectMapper(this.factory) : new ObjectMapper()); } configure(mapper); return (T) mapper; }
Configure an existing ObjectMapper instance with this builder's settings. This can be applied to any number of ObjectMappers.
Params:
  • objectMapper – the ObjectMapper to configure
/** * Configure an existing {@link ObjectMapper} instance with this builder's * settings. This can be applied to any number of {@code ObjectMappers}. * @param objectMapper the ObjectMapper to configure */
public void configure(ObjectMapper objectMapper) { Assert.notNull(objectMapper, "ObjectMapper must not be null"); if (this.findModulesViaServiceLoader) { objectMapper.registerModules(ObjectMapper.findModules(this.moduleClassLoader)); } else if (this.findWellKnownModules) { registerWellKnownModulesIfAvailable(objectMapper); } if (this.modules != null) { objectMapper.registerModules(this.modules); } if (this.moduleClasses != null) { for (Class<? extends Module> module : this.moduleClasses) { objectMapper.registerModule(BeanUtils.instantiateClass(module)); } } if (this.dateFormat != null) { objectMapper.setDateFormat(this.dateFormat); } if (this.locale != null) { objectMapper.setLocale(this.locale); } if (this.timeZone != null) { objectMapper.setTimeZone(this.timeZone); } if (this.annotationIntrospector != null) { objectMapper.setAnnotationIntrospector(this.annotationIntrospector); } if (this.propertyNamingStrategy != null) { objectMapper.setPropertyNamingStrategy(this.propertyNamingStrategy); } if (this.defaultTyping != null) { objectMapper.setDefaultTyping(this.defaultTyping); } if (this.serializationInclusion != null) { objectMapper.setSerializationInclusion(this.serializationInclusion); } if (this.filters != null) { objectMapper.setFilterProvider(this.filters); } this.mixIns.forEach(objectMapper::addMixIn); if (!this.serializers.isEmpty() || !this.deserializers.isEmpty()) { SimpleModule module = new SimpleModule(); addSerializers(module); addDeserializers(module); objectMapper.registerModule(module); } this.visibilities.forEach(objectMapper::setVisibility); customizeDefaultFeatures(objectMapper); this.features.forEach((feature, enabled) -> configureFeature(objectMapper, feature, enabled)); if (this.handlerInstantiator != null) { objectMapper.setHandlerInstantiator(this.handlerInstantiator); } else if (this.applicationContext != null) { objectMapper.setHandlerInstantiator( new SpringHandlerInstantiator(this.applicationContext.getAutowireCapableBeanFactory())); } } // Any change to this method should be also applied to spring-jms and spring-messaging // MappingJackson2MessageConverter default constructors private void customizeDefaultFeatures(ObjectMapper objectMapper) { if (!this.features.containsKey(MapperFeature.DEFAULT_VIEW_INCLUSION)) { configureFeature(objectMapper, MapperFeature.DEFAULT_VIEW_INCLUSION, false); } if (!this.features.containsKey(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES)) { configureFeature(objectMapper, DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false); } } @SuppressWarnings("unchecked") private <T> void addSerializers(SimpleModule module) { this.serializers.forEach((type, serializer) -> module.addSerializer((Class<? extends T>) type, (JsonSerializer<T>) serializer)); } @SuppressWarnings("unchecked") private <T> void addDeserializers(SimpleModule module) { this.deserializers.forEach((type, deserializer) -> module.addDeserializer((Class<T>) type, (JsonDeserializer<? extends T>) deserializer)); } private void configureFeature(ObjectMapper objectMapper, Object feature, boolean enabled) { if (feature instanceof JsonParser.Feature) { objectMapper.configure((JsonParser.Feature) feature, enabled); } else if (feature instanceof JsonGenerator.Feature) { objectMapper.configure((JsonGenerator.Feature) feature, enabled); } else if (feature instanceof SerializationFeature) { objectMapper.configure((SerializationFeature) feature, enabled); } else if (feature instanceof DeserializationFeature) { objectMapper.configure((DeserializationFeature) feature, enabled); } else if (feature instanceof MapperFeature) { objectMapper.configure((MapperFeature) feature, enabled); } else { throw new FatalBeanException("Unknown feature class: " + feature.getClass().getName()); } } @SuppressWarnings("unchecked") private void registerWellKnownModulesIfAvailable(ObjectMapper objectMapper) { try { Class<? extends Module> jdk8Module = (Class<? extends Module>) ClassUtils.forName("com.fasterxml.jackson.datatype.jdk8.Jdk8Module", this.moduleClassLoader); objectMapper.registerModule(BeanUtils.instantiateClass(jdk8Module)); } catch (ClassNotFoundException ex) { // jackson-datatype-jdk8 not available } try { Class<? extends Module> javaTimeModule = (Class<? extends Module>) ClassUtils.forName("com.fasterxml.jackson.datatype.jsr310.JavaTimeModule", this.moduleClassLoader); objectMapper.registerModule(BeanUtils.instantiateClass(javaTimeModule)); } catch (ClassNotFoundException ex) { // jackson-datatype-jsr310 not available } // Joda-Time present? if (ClassUtils.isPresent("org.joda.time.LocalDate", this.moduleClassLoader)) { try { Class<? extends Module> jodaModule = (Class<? extends Module>) ClassUtils.forName("com.fasterxml.jackson.datatype.joda.JodaModule", this.moduleClassLoader); objectMapper.registerModule(BeanUtils.instantiateClass(jodaModule)); } catch (ClassNotFoundException ex) { // jackson-datatype-joda not available } } // Kotlin present? if (KotlinDetector.isKotlinPresent()) { try { Class<? extends Module> kotlinModule = (Class<? extends Module>) ClassUtils.forName("com.fasterxml.jackson.module.kotlin.KotlinModule", this.moduleClassLoader); objectMapper.registerModule(BeanUtils.instantiateClass(kotlinModule)); } catch (ClassNotFoundException ex) { if (!kotlinWarningLogged) { kotlinWarningLogged = true; logger.warn("For Jackson Kotlin classes support please add " + "\"com.fasterxml.jackson.module:jackson-module-kotlin\" to the classpath"); } } } } // Convenience factory methods
Obtain a Jackson2ObjectMapperBuilder instance in order to build a regular JSON ObjectMapper instance.
/** * Obtain a {@link Jackson2ObjectMapperBuilder} instance in order to * build a regular JSON {@link ObjectMapper} instance. */
public static Jackson2ObjectMapperBuilder json() { return new Jackson2ObjectMapperBuilder(); }
Obtain a Jackson2ObjectMapperBuilder instance in order to build an XmlMapper instance.
/** * Obtain a {@link Jackson2ObjectMapperBuilder} instance in order to * build an {@link XmlMapper} instance. */
public static Jackson2ObjectMapperBuilder xml() { return new Jackson2ObjectMapperBuilder().createXmlMapper(true); }
Obtain a Jackson2ObjectMapperBuilder instance in order to build a Smile data format ObjectMapper instance.
Since:5.0
/** * Obtain a {@link Jackson2ObjectMapperBuilder} instance in order to * build a Smile data format {@link ObjectMapper} instance. * @since 5.0 */
public static Jackson2ObjectMapperBuilder smile() { return new Jackson2ObjectMapperBuilder().factory(new SmileFactoryInitializer().create()); }
Obtain a Jackson2ObjectMapperBuilder instance in order to build a CBOR data format ObjectMapper instance.
Since:5.0
/** * Obtain a {@link Jackson2ObjectMapperBuilder} instance in order to * build a CBOR data format {@link ObjectMapper} instance. * @since 5.0 */
public static Jackson2ObjectMapperBuilder cbor() { return new Jackson2ObjectMapperBuilder().factory(new CborFactoryInitializer().create()); } private static class XmlObjectMapperInitializer { public ObjectMapper create() { return new XmlMapper(StaxUtils.createDefensiveInputFactory()); } public ObjectMapper create(boolean defaultUseWrapper) { JacksonXmlModule module = new JacksonXmlModule(); module.setDefaultUseWrapper(defaultUseWrapper); return new XmlMapper(new XmlFactory(StaxUtils.createDefensiveInputFactory()), module); } } private static class SmileFactoryInitializer { public JsonFactory create() { return new SmileFactory(); } } private static class CborFactoryInitializer { public JsonFactory create() { return new CBORFactory(); } } }