Spring / Rest @PathVariable кодировка символов - PullRequest
26 голосов
/ 17 декабря 2010

В среде, которую я использую (Tomcat 6), последовательности процентов в сегментах пути, по-видимому, декодируются с использованием ISO-8859-1 при сопоставлении с @ PathVariable.

Я бы хотел, чтобы это былоUTF-8.

Я уже настроил Tomcat для использования UTF-8 (используя атрибут URIEncoding в server.xml).

Spring / Rest выполняет декодирование самостоятельно?Если да, где я могу переопределить кодировку по умолчанию?

Дополнительная информация;Вот мой тестовый код:

@RequestMapping( value = "/enc/{foo}", method = RequestMethod.GET )
public HttpEntity<String> enc( @PathVariable( "foo" ) String foo, HttpServletRequest req )
{
  String resp;

  resp = "      path variable foo: " + foo + "\n" + 
         "      req.getPathInfo(): " + req.getPathInfo() + "\n" +
         "req.getPathTranslated(): " + req.getPathTranslated() + "\n" + 
         "    req.getRequestURI(): " + req.getRequestURI() + "\n" + 
         "   req.getContextPath(): " + req.getContextPath() + "\n";

  HttpHeaders headers = new HttpHeaders();
  headers.setContentType( new MediaType( "text", "plain", Charset.forName( "UTF-8" ) ) );
  return new HttpEntity<String>( resp, headers );
}

Если я выполняю HTTP-запрос GET со следующим путем URI:

/TEST/enc/%c2%a3%20and%20%e2%82%ac%20rates

, который представляет собой кодировку UTF-8, а затем кодированную в процентах форму

/TEST/enc/£ and € rates

вывод, который я получаю:

      path variable foo: £ and ⬠rates
      req.getPathInfo(): /enc/£ and € rates
req.getPathTranslated(): C:\Users\jre\workspace\.metadata\.plugins\org.eclipse.wst.server.core\tmp0\wtpwebapps\TEST\enc\£ and € rates
    req.getRequestURI(): /TEST/enc/%C2%A3%20and%20%E2%82%AC%20rates
   req.getContextPath(): /TEST

, который мне показывает, что Tomcat (после установки атрибута URIEncoding) делает правильные вещи (см. GetPathInfo ()), нопеременная пути все еще декодируется в ISO-8859-1.

И ответ :

Spring / Rest, по-видимому, использует кодировку запроса, что является очень странной вещьюделать, поскольку речь идет о теле , а не об URI.Вздох.

Добавление этого:

<filter>
  <filter-name>CharacterEncodingFilter</filter-name>
  <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
  <init-param>
      <param-name>encoding</param-name>
      <param-value>UTF-8</param-value>
  </init-param>
</filter>
<filter-mapping>
  <filter-name>CharacterEncodingFilter</filter-name>
  <url-pattern>/*</url-pattern>
</filter-mapping>

решило проблему.Это действительно должно быть проще.

И на самом деле, это еще хуже:

Если метод действительно имеет тело запроса, и это не 't закодирован в UTF-8, требуется дополнительный параметр forceEncoding.Кажется, это работает, но я обеспокоен тем, что позже это вызовет больше проблем.

Другой подход

Тем временем я обнаружил, что можно отключитьрасшифровка, мое указание

<property name="urlDecode" value="false"/>

... в этом случае получатель может правильно ;но, конечно, это усложнит многие другие вещи.

Ответы [ 4 ]

27 голосов
/ 17 декабря 2010

Мне нужно добавить фильтр в web.xml

<filter>
    <filter-name>CharacterEncodingFilter</filter-name>
    <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
    <init-param>
        <param-name>encoding</param-name>
        <param-value>UTF-8</param-value>
    </init-param>
    <init-param>
        <param-name>forceEncoding</param-name>
        <param-value>true</param-value>
    </init-param>
</filter>
<filter-mapping>
    <filter-name>CharacterEncodingFilter</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>
4 голосов
/ 27 февраля 2012

Переменная пути все еще декодируется в ISO-8859-1 для меня, даже с фильтром кодировки символов. Вот что я должен был сделать, чтобы обойти это. Пожалуйста, дайте мне знать, если у вас есть другие идеи!

Чтобы увидеть фактические UTF-8-декодированные символы на сервере, вы можете просто сделать это и взглянуть на значение (вам нужно добавить «HttpServletRequest httpServletRequest» к параметрам вашего контроллера):

String requestURI = httpServletRequest.getRequestURI();
String decodedURI = URLDecoder.decode(requestURI, "UTF-8");

Затем я могу делать все, что захочу (например, получить параметр вручную из декодированного URI), теперь, когда у меня есть правильные декодированные данные на сервере.

3 голосов
/ 26 ноября 2016

Попробуйте настроить соединитель на Tomcat в server.xml. Добавьте useBodyEncodingForURI="true" или URIEncoding="UTF-8" к тегу Connector. Например:

    <Connector port="8080" protocol="HTTP/1.1"
           connectionTimeout="20000"
           useBodyEncodingForURI="true"
           redirectPort="8443" />
0 голосов
/ 10 июня 2014

Но разве не ужасно, что вам вообще нужно связываться с конфигурацией Tomcat (URIEncoding), чтобы это работало?Если API сервлета предоставляет способ получения пути и параметров запроса в их некодированном представлении, приложение (или Spring) может полностью справиться с декодированием.И, очевидно, HttpServletRequest#getPathInfo и HttpServletRequest#getQueryString могли бы даже обеспечить это, но для последнего это означало бы, что Spring должен был бы анализировать и декодировать саму строку запроса, а не полагаться на HttpServletRequest#getParameter и друзей.Очевидно, они этого не делают, а это значит, что вы не можете @RequestParam или @PathVariable перехватывать что-либо, кроме строк us-ascii, безопасно, не полагаясь на конфигурацию контейнера сервлета.

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