Spring RestTemplate HTTP Post с параметрами вызывает 400 плохую ошибку запроса

возможные дубликаты нужна помощь на запросе столба RestTemplate с параметрами тела? и Весна RESTtemplate сообщение но эти ответы не работает для меня
Я попытался получить токен доступа от Instagram API от Spring Android. Просьба от документ Instagram такой:

curl -F 'client_id=CLIENT-ID' 
-F 'client_secret=CLIENT-SECRET' 
-F 'grant_type=authorization_code' 
-F 'redirect_uri=YOUR-REDIRECT-URI' 
-F 'code=CODE' https://api.instagram.com/oauth/access_token

вот мой токен доступа к запросу (после того, как я получу успешный токен запроса):

 MultiValueMap<String, String> mvm = new LinkedMultiValueMap<String, String>();
    mvm.add("client_id", INSTAGRAM_CILENT_ID);
    mvm.add("client_secret", INSTAGRAM_SECRET);
    mvm.add("grant_type", "authorization_code");
    mvm.add("redirect_uri", CALLBACKURL);
    mvm.add("code", requestToken);

    InstagramResult result = restTemplate .postForObject("https://api.instagram.com/oauth/access_token", mvm, InstagramResult .class);

отображение результата класс:

public class InstagramLogin {
   public String access_token;
   public InstagramUser user;
}

public class InstagramUser {
   public String id;
   public String username;
   public String full_name;
   public String profile_picture;
}

и остальные шаблон:

   RestTemplate restTemplate = new RestTemplate();
    final List<HttpMessageConverter<?>> listHttpMessageConverters = new ArrayList< HttpMessageConverter<?> >(); 

    listHttpMessageConverters.add(new GsonHttpMessageConverter());
    listHttpMessageConverters.add(new FormHttpMessageConverter());
    listHttpMessageConverters.add(new StringHttpMessageConverter());
    restTemplate.setMessageConverters(listHttpMessageConverters);

но я всегда получаю ошибка 400: неверный запрос. Вот моя трассировка стека:

04-03 09:32:45.366: W/RestTemplate(31709): POST request for "https://api.instagram.com/oauth/access_token" resulted in 400 (BAD REQUEST); invoking error handler
04-03 09:32:46.857: E//DefaultRequestRunner.java:138(31709): 09:32:46.862 Thread-32787 An exception occurred during request network execution :400 BAD REQUEST
04-03 09:32:46.857: E//DefaultRequestRunner.java:138(31709): org.springframework.web.client.HttpClientErrorException: 400 BAD REQUEST
04-03 09:32:46.857: E//DefaultRequestRunner.java:138(31709):    at org.springframework.web.client.DefaultResponseErrorHandler.handleError(DefaultResponseErrorHandler.java:76)
04-03 09:32:46.857: E//DefaultRequestRunner.java:138(31709):    at org.springframework.web.client.RestTemplate.handleResponseError(RestTemplate.java:524)
04-03 09:32:46.857: E//DefaultRequestRunner.java:138(31709):    at org.springframework.web.client.RestTemplate.doExecute(RestTemplate.java:481)
04-03 09:32:46.857: E//DefaultRequestRunner.java:138(31709):    at org.springframework.web.client.RestTemplate.execute(RestTemplate.java:439)
04-03 09:32:46.857: E//DefaultRequestRunner.java:138(31709):    at org.springframework.web.client.RestTemplate.postForObject(RestTemplate.java:317)

P / s: Я уверен, что мои значения параметров верны, потому что я тестировал curl, и он работал нормально.

2 ответов


сервер часто возвращает HTTP 400, если тип контента не приемлем для запроса. Пример curl из instagram использует -F параметр, указывающий многостраничные данные post:

-F, --form CONTENT  Specify HTTP multipart POST data (H)

таким образом, вы можете попробовать явно установить HTTP-заголовок Content-type в запросе RestTemplate:

HttpHeaders requestHeaders = new HttpHeaders();
requestHeaders.setContentType(MediaType.MULTIPART_FORM_DATA);
HttpEntity<MultiValueMap<String, String>> requestEntity = new HttpEntity<MultiValueMap<String, String>>(mvm, requestHeaders);
ResponseEntity<InstagramResult> response = restTemplate.exchange(url, HttpMethod.POST, requestEntity, InstagramResult.class);
InstagramResult result = response.getBody();

как упоминалось ранее в комментариях, прокси-инструмент, такой как fiddler, может быть действительно полезен для отладки. Проблема в этой ситуации что вы работаете с SSL, поэтому эти инструменты не смогут "видеть" зашифрованные сообщения без специальной конфигурации.


HttpClient httpclient = new DefaultHttpClient();
HttpPost httppost = new HttpPost(LINKEDIN_POST_URL);

List<NameValuePair> nameValuePairs = new ArrayList<NameValuePair>(2);
nameValuePairs.add(new BasicNameValuePair("grant_type", grant_type));
nameValuePairs.add(new BasicNameValuePair("code", code));
nameValuePairs.add(new BasicNameValuePair("redirect_uri", redirect_uri);
nameValuePairs.add(new BasicNameValuePair("client_id", client_id));
nameValuePairs.add(new BasicNameValuePair("client_secret", client_secret));

httppost.setEntity(new UrlEncodedFormEntity(nameValuePairs));
HttpResponse response = httpclient.execute(httppost);

и преобразовать ваш ответ в строку. Вы можете использовать:

EntityUtils.toString(response.getEntity());

Я полагаю, вы хотите отправить запрос post для любого объекта. Для этого используется DefaultHttpClient. HttpPost объект будет принимать url в качестве аргумента (только url не параметры). Параметры, которые вы хотите отправить, можно редактировать выше в объекте BasicNameValuePair. Первый параметр в качестве имени параметра, а второй - его значение. Как только вы вызовете метод execute, ответ придет в объект HttpResponse. Теперь вам нужно преобразовать объект HttpResponse в Entity. И из сущности вы можете вызвать EntityUtils.метод toString () для преобразования в String. Полагаю, вы ожидаете данные json. Возможно, вам придется сопоставить соответствующий ему класс для преобразования данных json в объект класса java.