All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.springframework.web.servlet.config.annotation.ContentNegotiationConfigurer Maven / Gradle / Ivy

There is a newer version: 6.1.6
Show newest version
/*
 * Copyright 2002-2020 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
 *
 *      https://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.springframework.web.servlet.config.annotation;

import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import javax.servlet.ServletContext;

import org.springframework.http.MediaType;
import org.springframework.http.MediaTypeFactory;
import org.springframework.lang.Nullable;
import org.springframework.web.accept.ContentNegotiationManager;
import org.springframework.web.accept.ContentNegotiationManagerFactoryBean;
import org.springframework.web.accept.ContentNegotiationStrategy;
import org.springframework.web.accept.FixedContentNegotiationStrategy;
import org.springframework.web.accept.HeaderContentNegotiationStrategy;
import org.springframework.web.accept.ParameterContentNegotiationStrategy;

/**
 * Creates a {@code ContentNegotiationManager} and configures it with
 * one or more {@link ContentNegotiationStrategy} instances.
 *
 * 

This factory offers properties that in turn result in configuring the * underlying strategies. The table below shows the property names, their * default settings, as well as the strategies that they help to configure: * *

* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
Property SetterDefault ValueUnderlying StrategyEnabled Or Not
{@link #favorParameter}false{@link ParameterContentNegotiationStrategy}Off
{@link #favorPathExtension}false (as of 5.3){@link org.springframework.web.accept.PathExtensionContentNegotiationStrategy * PathExtensionContentNegotiationStrategy}Off
{@link #ignoreAcceptHeader}false{@link HeaderContentNegotiationStrategy}Enabled
{@link #defaultContentType}null{@link FixedContentNegotiationStrategy}Off
{@link #defaultContentTypeStrategy}null{@link ContentNegotiationStrategy}Off
* *

As of 5.0 you can set the exact strategies to use via * {@link #strategies(List)}. * *

Note: if you must use URL-based content type resolution, * the use of a query parameter is simpler and preferable to the use of a path * extension since the latter can cause issues with URI variables, path * parameters, and URI decoding. Consider setting {@link #favorPathExtension} * to {@literal false} or otherwise set the strategies to use explicitly via * {@link #strategies(List)}. * * @author Rossen Stoyanchev * @since 3.2 */ public class ContentNegotiationConfigurer { private final ContentNegotiationManagerFactoryBean factory = new ContentNegotiationManagerFactoryBean(); private final Map mediaTypes = new HashMap<>(); /** * Class constructor with {@link javax.servlet.ServletContext}. */ public ContentNegotiationConfigurer(@Nullable ServletContext servletContext) { if (servletContext != null) { this.factory.setServletContext(servletContext); } } /** * Set the exact list of strategies to use. *

Note: use of this method is mutually exclusive with * use of all other setters in this class which customize a default, fixed * set of strategies. See class level doc for more details. * @param strategies the strategies to use * @since 5.0 */ public void strategies(@Nullable List strategies) { this.factory.setStrategies(strategies); } /** * Whether a request parameter ("format" by default) should be used to * determine the requested media type. For this option to work you must * register {@link #mediaType(String, MediaType) media type mappings}. *

By default this is set to {@code false}. * @see #parameterName(String) */ public ContentNegotiationConfigurer favorParameter(boolean favorParameter) { this.factory.setFavorParameter(favorParameter); return this; } /** * Set the query parameter name to use when {@link #favorParameter} is on. *

The default parameter name is {@code "format"}. */ public ContentNegotiationConfigurer parameterName(String parameterName) { this.factory.setParameterName(parameterName); return this; } /** * Whether the path extension in the URL path should be used to determine * the requested media type. *

By default this is set to {@code false} in which case path extensions * have no impact on content negotiation. * @deprecated as of 5.2.4. See deprecation note on * {@link ContentNegotiationManagerFactoryBean#setFavorPathExtension(boolean)}. */ @Deprecated public ContentNegotiationConfigurer favorPathExtension(boolean favorPathExtension) { this.factory.setFavorPathExtension(favorPathExtension); return this; } /** * Add a mapping from a key, extracted from a path extension or a query * parameter, to a MediaType. This is required in order for the parameter * strategy to work. Any extensions explicitly registered here are also * treated as safe for the purpose of Reflected File Download attack * detection (see Spring Framework reference documentation for more details * on RFD attack protection). *

The path extension strategy will also try to use * {@link ServletContext#getMimeType} and {@link MediaTypeFactory} to resolve path * extensions. To change this behavior see the {@link #useRegisteredExtensionsOnly} property. * @param extension the key to look up * @param mediaType the media type * @see #mediaTypes(Map) * @see #replaceMediaTypes(Map) */ public ContentNegotiationConfigurer mediaType(String extension, MediaType mediaType) { this.mediaTypes.put(extension, mediaType); return this; } /** * An alternative to {@link #mediaType}. * @see #mediaType(String, MediaType) * @see #replaceMediaTypes(Map) */ public ContentNegotiationConfigurer mediaTypes(@Nullable Map mediaTypes) { if (mediaTypes != null) { this.mediaTypes.putAll(mediaTypes); } return this; } /** * Similar to {@link #mediaType} but for replacing existing mappings. * @see #mediaType(String, MediaType) * @see #mediaTypes(Map) */ public ContentNegotiationConfigurer replaceMediaTypes(Map mediaTypes) { this.mediaTypes.clear(); mediaTypes(mediaTypes); return this; } /** * Whether to ignore requests with path extension that cannot be resolved * to any media type. Setting this to {@code false} will result in an * {@code HttpMediaTypeNotAcceptableException} if there is no match. *

By default this is set to {@code true}. * @deprecated as of 5.2.4. See deprecation note on * {@link ContentNegotiationManagerFactoryBean#setIgnoreUnknownPathExtensions(boolean)}. */ @Deprecated public ContentNegotiationConfigurer ignoreUnknownPathExtensions(boolean ignore) { this.factory.setIgnoreUnknownPathExtensions(ignore); return this; } /** * When {@link #favorPathExtension} is set, this property determines whether * to allow use of JAF (Java Activation Framework) to resolve a path * extension to a specific MediaType. * @deprecated as of 5.0, in favor of {@link #useRegisteredExtensionsOnly(boolean)} * which has reverse behavior */ @Deprecated public ContentNegotiationConfigurer useJaf(boolean useJaf) { return this.useRegisteredExtensionsOnly(!useJaf); } /** * When {@link #favorPathExtension favorPathExtension} is set, this * property determines whether to use only registered {@code MediaType} mappings * to resolve a path extension to a specific MediaType. *

By default this is not set in which case * {@code PathExtensionContentNegotiationStrategy} will use defaults if available. */ public ContentNegotiationConfigurer useRegisteredExtensionsOnly(boolean useRegisteredExtensionsOnly) { this.factory.setUseRegisteredExtensionsOnly(useRegisteredExtensionsOnly); return this; } /** * Whether to disable checking the 'Accept' request header. *

By default this value is set to {@code false}. */ public ContentNegotiationConfigurer ignoreAcceptHeader(boolean ignoreAcceptHeader) { this.factory.setIgnoreAcceptHeader(ignoreAcceptHeader); return this; } /** * Set the default content type(s) to use when no content type is requested * in order of priority. *

If destinations are present that do not support any of the given media * types, consider appending {@link MediaType#ALL} at the end. *

By default this is not set. * @see #defaultContentTypeStrategy */ public ContentNegotiationConfigurer defaultContentType(MediaType... defaultContentTypes) { this.factory.setDefaultContentTypes(Arrays.asList(defaultContentTypes)); return this; } /** * Set a custom {@link ContentNegotiationStrategy} to use to determine * the content type to use when no content type is requested. *

By default this is not set. * @since 4.1.2 * @see #defaultContentType */ public ContentNegotiationConfigurer defaultContentTypeStrategy(ContentNegotiationStrategy defaultStrategy) { this.factory.setDefaultContentTypeStrategy(defaultStrategy); return this; } /** * Build a {@link ContentNegotiationManager} based on this configurer's settings. * @since 4.3.12 * @see ContentNegotiationManagerFactoryBean#getObject() */ protected ContentNegotiationManager buildContentNegotiationManager() { this.factory.addMediaTypes(this.mediaTypes); return this.factory.build(); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy