Выполнение POST-вызова вместо GET с использованием urllib2
Есть много вещей по поводу вызовов urllib2 и POST, но я застрял в проблеме.
Я пытаюсь сделать простой вызов POST для службы:
url = 'http://myserver/post_service'
data = urllib.urlencode({'name' : 'joe',
'age' : '10'})
content = urllib2.urlopen(url=url, data=data).read()
print content
Я вижу журналы сервера, и он говорит, что я выполняю GET-вызовы, когда отправляю аргумент данных в urlopen.
Библиотека выдает ошибку 404 (не найдена), что правильно для вызова GET, вызовы POST обрабатываются хорошо (я также пытаюсь использовать POST в форме HTML).
7 ответов
На этот вопрос можно было ответить раньше: Python URLLib / URLLib2 POST.
Ваш сервер, вероятно, выполняет перенаправление 302 с http://myserver/post_service
в http://myserver/post_service/
, Когда перенаправление 302 выполняется, запрос изменяется с POST на GET (см. Выпуск 1401). Попробуйте изменить url
в http://myserver/post_service/
,
Сделайте это поэтапно и измените объект следующим образом:
# make a string with the request type in it:
method = "POST"
# create a handler. you can specify different handlers here (file uploads etc)
# but we go for the default
handler = urllib2.HTTPHandler()
# create an openerdirector instance
opener = urllib2.build_opener(handler)
# build a request
data = urllib.urlencode(dictionary_of_POST_fields_or_None)
request = urllib2.Request(url, data=data)
# add any other information you want
request.add_header("Content-Type",'application/json')
# overload the get method function with a small anonymous function...
request.get_method = lambda: method
# try it; don't forget to catch the result
try:
connection = opener.open(request)
except urllib2.HTTPError,e:
connection = e
# check. Substitute with appropriate HTTP code.
if connection.code == 200:
data = connection.read()
else:
# handle the error case. connection.read() will still contain data
# if any was returned, but it probably won't be of any use
Этот способ позволяет вам продлить PUT
, DELETE
, HEAD
а также OPTIONS
запросы тоже, просто подставляя значение метода или даже заключая его в функцию. В зависимости от того, что вы пытаетесь сделать, вам также может понадобиться другой обработчик HTTP, например, для загрузки нескольких файлов.
Модуль запросов может облегчить вашу боль.
url = 'http://myserver/post_service'
data = dict(name='joe', age='10')
r = requests.post(url, data=data, allow_redirects=True)
print r.content
Прочитайте Руководство по отсутствию urllib. Извлечен оттуда следующий простой пример запроса POST.
url = 'http://myserver/post_service'
data = urllib.urlencode({'name' : 'joe', 'age' : '10'})
req = urllib2.Request(url, data)
response = urllib2.urlopen(req)
print response.read()
По предложению @Michael Kent учитывает запросы, это здорово.
РЕДАКТИРОВАТЬ: При этом я не знаю, почему передача данных в urlopen() не приводит к запросу POST; Должно. Я подозреваю, что ваш сервер перенаправляет или неправильно работает.
Он должен отправлять POST, если вы предоставляете параметр данных (как вы делаете):
из документации: "HTTP-запрос будет POST вместо GET, если указан параметр data"
так что... добавьте отладочный вывод, чтобы увидеть, что происходит со стороны клиента.
Вы можете изменить свой код к этому и попробовать еще раз:
import urllib
import urllib2
url = 'http://myserver/post_service'
opener = urllib2.build_opener(urllib2.HTTPHandler(debuglevel=1))
data = urllib.urlencode({'name' : 'joe',
'age' : '10'})
content = opener.open(url, data=data).read()
Попробуйте это вместо этого:
url = 'http://myserver/post_service'
data = urllib.urlencode({'name' : 'joe',
'age' : '10'})
req = urllib2.Request(url=url,data=data)
content = urllib2.urlopen(req).read()
print content
url="https://myserver/post_service"
data["name"] = "joe"
data["age"] = "20"
data_encoded = urllib2.urlencode(data)
print urllib2.urlopen(url + "?" + data_encoded).read()
Может быть, это может помочь