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.