«+» (Знак плюс), не кодируются с использованием RestTemplate Строка URL, но интерпретируется как «» (пробел)

Вопрос задан: 1 год назад Последняя активность: 1 год назад
up 14 down

Мы переходим от Java 8, Java 11, и, таким образом, с весны Ботинок 1.5.6 до 2.1.2. Мы заметили, что при использовании RestTemplate, знак «+» не кодируется «% 2B» больше (изменения по SPR-14828). Это было бы хорошо, потому что RFC3986 не список «+» как зарезервированное характер, но она по-прежнему интерпретируется как «» (пробел) при получении в Спринг загрузки конечной точки.

У нас есть поисковый запрос, который может принимать дополнительные временные метки в качестве параметров запроса. Запрос выглядит что-то вроде http://example.com/search?beforeTimestamp=2019-01-21T14:56:50%2B00:00.

Мы не можем понять, как отправить закодированный знак плюс, не будучи дважды закодирован. параметр запроса 2019-01-21T14:56:50+00:00 будет интерпретироваться как 2019-01-21T14:56:50 00:00. Если бы мы должны были кодировать параметр сами (2019-01-21T14:56:50%2B00:00), То она будет получена, и интерпретировать как 2019-01-21T14:56:50%252B00:00.

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

Кроме того, есть ли способ заставить «+» не следует интерпретировать как «» конечную точку?

Я написал краткий пример демонстрируют некоторые способы достижения строже кодирования с их недостатками объяснены в качестве комментариев:

package com.example.clientandserver;

import org.springframework.boot.CommandLineRunner;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.web.client.RestTemplateBuilder;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.client.RestTemplate;
import org.springframework.web.util.DefaultUriBuilderFactory;
import org.springframework.web.util.UriComponentsBuilder;
import org.springframework.web.util.UriUtils;

import java.nio.charset.StandardCharsets;
import java.util.HashMap;
import java.util.Map;

@SpringBootApplication
@RestController
public class ClientAndServerApp implements CommandLineRunner {

    public static void main(String[] args) {
        SpringApplication.run(ClientAndServerApp.class, args);
    }

    @Override
    public void run(String... args) {
        String beforeTimestamp = "2019-01-21T14:56:50+00:00";

        // Previously - base url and raw params (encoded automatically). This worked in the earlier version of Spring Boot
        {
            RestTemplate restTemplate = new RestTemplateBuilder().rootUri("http://localhost:8080").build();
            UriComponentsBuilder b = UriComponentsBuilder.fromPath("/search");
            if (beforeTimestamp != null) {
                b.queryParam("beforeTimestamp", beforeTimestamp);
            }
            restTemplate.getForEntity(b.toUriString(), Object.class);
            // Received: 2019-01-21T14:56:50 00:00
            //       Plus sign missing here ^
        }

        // Option 1 - no base url and encoding the param ourselves.
        {
            RestTemplate restTemplate = new RestTemplate();
            UriComponentsBuilder b = UriComponentsBuilder.fromHttpUrl("http://localhost:8080/search");
            if (beforeTimestamp != null) {
                b.queryParam("beforeTimestamp", UriUtils.encode(beforeTimestamp, StandardCharsets.UTF_8));
            }
            restTemplate.getForEntity(b.build(true).toUri(), Object.class).getBody();
            // Received: 2019-01-21T14:56:50+00:00
        }

        // Option 2 - with base url, but templated, so query parameter is not optional.
        {
            RestTemplate restTemplate = new RestTemplateBuilder().rootUri("http://localhost:8080").uriTemplateHandler(new DefaultUriBuilderFactory()).build();
            Map<String, String> params = new HashMap<>();
            params.put("beforeTimestamp", beforeTimestamp);
            restTemplate.getForEntity("/search?beforeTimestamp={beforeTimestamp}", Object.class, params);
            // Received: 2019-01-21T14:56:50+00:00
        }
    }

    @GetMapping("/search")
    public void search(@RequestParam String beforeTimestamp) {
        System.out.println("Received: " + beforeTimestamp);
    }
}

1 ответ

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

Реклама

up 7 down accepted

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

RestTemplate restTemplate = new RestTemplateBuilder()
        .rootUri("http://localhost:8080")
        .interceptors(new PlusEncoderInterceptor())
        .build();

Сокращенный пример:

public class PlusEncoderInterceptor implements ClientHttpRequestInterceptor {

    @Override
    public ClientHttpResponse intercept(HttpRequest request, byte[] body, ClientHttpRequestExecution execution) throws IOException {
        return execution.execute(new HttpRequestWrapper(request) {
            @Override
            public URI getURI() {
                URI u = super.getURI();
                String strictlyEscapedQuery = StringUtils.replace(u.getRawQuery(), "+", "%2B");
                return UriComponentsBuilder.fromUri(u)
                        .replaceQuery(strictlyEscapedQuery)
                        .build(true).toUri();
            }
        }, body);
    }
}

Ошибка 505

Что-то пошло не так

Попробуйте воспользоваться поиском