it-roy-ru.com

Spring MVC @PathVariable с точкой (.) Усекается

Это продолжение вопроса Spring MVC @PathVariable усекается

Весенний форум заявляет, что он исправил (версия 3.2) как часть ContentNegotiationManager. см. ссылку ниже.
https://jira.springsource.org/browse/SPR-6164
https://jira.springsource.org/browse/SPR-7632

В моем приложении requestParameter с .com усекается.

Может ли кто-нибудь объяснить мне, как использовать эту новую функцию? как это настраивается в XML?

Примечание: spring forum- # 1 Spring MVC @PathVariable с точкой (.) Усекается

311
Kanagavelu Sugumar

Насколько я знаю, эта проблема возникает только для пути переменной в конце запроса сопоставления. 

Мы смогли решить эту проблему, определив дополнение регулярных выражений в сопоставлении запросов.

 /somepath/{variable:.+}
429
Martin Frey

Spring считает, что все, что находится за последней точкой, является расширением файла, например .jsonor .xml, и использует его для получения вашего параметра.

Так что если у вас есть /somepath/{variable}:

  • /somepath/param, /somepath/param.json, /somepath/param.xml или /somepath/param.anything приведут к параметру со значением param
  • /somepath/param.value.json, /somepath/param.value.xml или /somepath/param.value.anything приведут к параметру со значением param.value

если вы измените свое отображение на /somepath/{variable:.+}, как предложено, любая точка, включая последнюю, будет считаться частью вашего параметра:

  • /somepath/param приведет к параметру со значением param
  • /somepath/param.json приведет к параметру со значением param.json
  • /somepath/param.xml приведет к параметру со значением param.xml 
  • /somepath/param.anything приведет к параметру со значением param.anything
  • /somepath/param.value.json приведет к параметру со значением param.value.json
  • ...

Если вас не волнует распознавание расширений, вы можете отключить его, переопределив mvc:annotation-driven automagic:

<bean id="handlerMapping"
      class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping">
    <property name="contentNegotiationManager" ref="contentNegotiationManager"/>
    <property name="useSuffixPatternMatch" value="false"/>
</bean>

Итак, еще раз, если у вас есть /somepath/{variable}:

  • /somepath/param, /somepath/param.json, /somepath/param.xml или /somepath/param.anything приведут к параметру со значением param
  • /somepath/param.value.json, /somepath/param.value.xml или /somepath/param.value.anything приведут к параметру со значением param.value

примечание: отличие от конфигурации по умолчанию видно только в том случае, если у вас есть отображение типа somepath/something.{variable}. см. выпуск проекта Resthub

если вы хотите сохранить управление расширениями, начиная с Spring 3.2, вы также можете установить свойство useRegisteredSuffixPatternMatch компонента bean RequestMappingHandlerMapping, чтобы сохранить активацию распознавания суффиксаPattern, но только для зарегистрированного расширения.

Здесь вы определяете только расширения json и xml: 

<bean id="handlerMapping"
      class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping">
    <property name="contentNegotiationManager" ref="contentNegotiationManager"/>
    <property name="useRegisteredSuffixPatternMatch" value="true"/>
</bean>

<bean id="contentNegotiationManager" class="org.springframework.web.accept.ContentNegotiationManagerFactoryBean">
    <property name="favorPathExtension" value="false"/>
    <property name="favorParameter" value="true"/>
    <property name="mediaTypes">
        <value>
            json=application/json
            xml=application/xml
        </value>
    </property>
</bean>

Обратите внимание, что mvc: annotation-driven принимает теперь опцию contentNegotiation для предоставления пользовательского компонента, но свойство RequestMappingHandlerMapping должно быть изменено на true (по умолчанию false) (ср. https://jira.springsource.org/browse/SPR -7632 ).

По этой причине вам все равно придется переопределить все настройки mvc: annotation. Я открыл билет в Spring, чтобы попросить пользовательский RequestMappingHandlerMapping: https://jira.springsource.org/browse/SPR-11253 . Пожалуйста, проголосуйте, если вы заинтересованы в.

При переопределении будьте внимательны, если учесть также переопределение пользовательского управления выполнением. В противном случае все ваши пользовательские сопоставления исключений потерпят неудачу. Вам придется повторно использовать MessageCoverters с bean-компонентом списка:

<bean id="validator" class="org.springframework.validation.beanvalidation.LocalValidatorFactoryBean" />
<bean id="conversionService" class="org.springframework.format.support.FormattingConversionServiceFactoryBean" />

<util:list id="messageConverters">
    <bean class="your.custom.message.converter.IfAny"></bean>
    <bean class="org.springframework.http.converter.ByteArrayHttpMessageConverter"></bean>
    <bean class="org.springframework.http.converter.StringHttpMessageConverter"></bean>
    <bean class="org.springframework.http.converter.ResourceHttpMessageConverter"></bean>
    <bean class="org.springframework.http.converter.xml.SourceHttpMessageConverter"></bean>
    <bean class="org.springframework.http.converter.xml.XmlAwareFormHttpMessageConverter"></bean>
    <bean class="org.springframework.http.converter.xml.Jaxb2RootElementHttpMessageConverter"></bean>
    <bean class="org.springframework.http.converter.json.MappingJacksonHttpMessageConverter"></bean>
</util:list>

<bean name="exceptionHandlerExceptionResolver"
      class="org.springframework.web.servlet.mvc.method.annotation.ExceptionHandlerExceptionResolver">
    <property name="order" value="0"/>
    <property name="messageConverters" ref="messageConverters"/>
</bean>

<bean name="handlerAdapter"
      class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter">
    <property name="webBindingInitializer">
        <bean class="org.springframework.web.bind.support.ConfigurableWebBindingInitializer">
            <property name="conversionService" ref="conversionService" />
            <property name="validator" ref="validator" />
        </bean>
    </property>
    <property name="messageConverters" ref="messageConverters"/>
</bean>

<bean id="handlerMapping"
      class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping">
</bean>

Я реализовал в проекте с открытым исходным кодом Resthub , частью которого я являюсь, набор тестов по этим предметам: см. https://github.com/resthub/resthub-spring-stack/pull/219/ файлы & https://github.com/resthub/resthub-spring-stack/issues/217

220
bmeurant

Обновление для Spring 4: начиная с 4.0.1 вы можете использовать PathMatchConfigurer (через вашу WebMvcConfigurer), например,.

@Configuration
protected static class AllResources extends WebMvcConfigurerAdapter {

    @Override
    public void configurePathMatch(PathMatchConfigurer matcher) {
        matcher.setUseRegisteredSuffixPatternMatch(true);
    }

}

В xml это будет ( https://jira.spring.io/browse/SPR-10163 ):

<mvc:annotation-driven>
    [...]
    <mvc:path-matching registered-suffixes-only="true"/>
</mvc:annotation-driven>
89
Dave Syer

В дополнение к ответу Мартина Фрея это также можно исправить, добавив косую черту в значение RequestMapping:

/path/{variable}/

Имейте в виду, что это исправление не поддерживает ремонтопригодность. Теперь требуется, чтобы все URI имели косую черту - что может быть неочевидно для пользователей API/новых разработчиков. Поскольку, вероятно, не все параметры могут содержать ., это также может приводить к периодическим ошибкам

83
Michał Rybak

добавление «:. +» работало на меня, но не до тех пор, пока я не удалил внешние фигурные скобки.

значение = {"/ имя пользователя/{id:. +}"} не работает

value = "/username/ndomid:.+}" работает

Надеюсь, я кому-то помог :)

25
Martin Čejka

В Spring Boot Rest Controller я решил эти проблемы следующим образом:

RestController:

@GetMapping("/statusByEmail/{email:.+}/")
public String statusByEmail(@PathVariable(value = "email") String email){
  //code
}

И От Отдыха Клиента:

Get http://mywebhook.com/statusByEmail/[email protected]/
24
GoutamS

/somepath/{variable:.+} работает в теге Java requestMapping.

14
amit dahiya

Вот подход, основанный исключительно на конфигурации Java:

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurationSupport;
import org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping;

@Configuration
public class MvcConfig extends WebMvcConfigurationSupport{

    @Bean
    public RequestMappingHandlerMapping requestMappingHandlerMapping() {
        RequestMappingHandlerMapping handlerMapping = super.requestMappingHandlerMapping();
        handlerMapping.setUseSuffixPatternMatch(false);
        handlerMapping.setUseTrailingSlashMatch(false);
        return handlerMapping;
    }
}
12
Bruno Carrier

Один из довольно простых способов обойти эту проблему - добавить косую черту ...

например.:

использовать:

/somepath/filename.jpg/

вместо:

/somepath/filename.jpg
10
Marcelo C.

В Spring Boot, регулярное выражение решает проблему, как

@GetMapping("/path/{param1:.+}")
9
Dan

Полное решение, включающее адреса электронной почты в именах путей для весны 4.2 

<bean id="contentNegotiationManager"
    class="org.springframework.web.accept.ContentNegotiationManagerFactoryBean">
    <property name="favorPathExtension" value="false" />
    <property name="favorParameter" value="true" />
    <property name="mediaTypes">
        <value>
            json=application/json
            xml=application/xml
        </value>
    </property>
</bean>
<mvc:annotation-driven
    content-negotiation-manager="contentNegotiationManager">
    <mvc:path-matching suffix-pattern="false" registered-suffixes-only="true" />
</mvc:annotation-driven>

Добавьте это в application-xml

6
Paul Arer

Если вы используете Spring 3.2.x и <mvc:annotation-driven />, создайте эту маленькую BeanPostProcessor:

package spring;

public final class DoNotTruncateMyUrls implements BeanPostProcessor {
    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        if (bean instanceof RequestMappingHandlerMapping) {
            ((RequestMappingHandlerMapping)bean).setUseSuffixPatternMatch(false);
        }
        return bean;
    }
    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        return bean;
    }
}

Затем поместите это в конфигурационный файл MVC xml:

<bean class="spring.DoNotTruncateMyUrls" />
4
Jukka

Наконец я нашел решение в Spring Docs :

Чтобы полностью отключить использование расширений файлов, необходимо установить оба следующих параметра: 

 useSuffixPatternMatching(false), see PathMatchConfigurer

 favorPathExtension(false), see ContentNegotiationConfigurer

Добавление этого к моей реализации WebMvcConfigurerAdapter решило проблему:

@Override
public void configureContentNegotiation(ContentNegotiationConfigurer configurer) {
    configurer.favorPathExtension(false);
}

@Override
public void configurePathMatch(PathMatchConfigurer matcher) {
    matcher.setUseSuffixPatternMatch(false);
}
1
luboskrnac

Для меня 

@GetMapping(path = "/a/{variableName:.+}")

работает, но только если вы также закодировали «точку» в URL-адресе запроса как «% 2E», тогда это работает. Но все URL должны быть такими ... которые не являются "стандартной" кодировкой, хотя и допустимы. Чувствуется что-то вроде ошибки: |

Другой обходной путь, похожий на способ «конечной косой черты», заключается в перемещении переменной, которая будет иметь точку «inline», например:

@GetMapping (path = "/ {variableName}/a")

теперь все точки будут сохранены, без изменений или регулярных выражений.

0
rogerdpack