Подтвердить что ты не робот

Выполнение вызова 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-форме).

Любые подсказки по этому поводу будут оценены

4b9b3361

Ответ 1

Об этом, возможно, ответили ранее: Python URLLib/URLLib2 POST.

Вероятно, ваш сервер выполняет перенаправление 302 от http://myserver/post_service до http://myserver/post_service/. Когда выполняется перенаправление 302, запрос изменяется с POST на GET (см. Issue 1401). Попробуйте изменить url на http://myserver/post_service/.

Ответ 2

Сделайте это поэтапно и измените объект, например:

# 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, например. для загрузки нескольких файлов.

Ответ 3

Модуль requests может облегчить вашу боль.

url = 'http://myserver/post_service'
data = dict(name='joe', age='10')

r = requests.post(url, data=data, allow_redirects=True)
print r.content

Ответ 4

Прочитайте urllib Missing Manual. Вытянутый оттуда следующий простой пример запроса 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, рассмотрите requests, это здорово.

EDIT: Это говорит о том, что я не знаю, почему передача данных в urlopen() не приводит к запросу POST; Должно. Я подозреваю, что ваш сервер перенаправляет или неправильно ведет себя.

Ответ 5

он должен отправлять POST, если вы предоставляете параметр данных (например, вы делаете):

из документов: "HTTP-запрос будет POST вместо GET, когда предоставлен параметр данных"

поэтому добавьте некоторый отладочный вывод, чтобы увидеть, что происходит с клиентской стороны.

вы можете изменить свой код и повторить попытку:

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()

Ответ 6

Попробуйте это вместо:

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

Ответ 7

url="https://myserver/post_service"
data["name"] = "joe"
data["age"] = "20"
data_encoded = urllib2.urlencode(data)
print urllib2.urlopen(url + "?" + data_encoded).read()

Возможно, это поможет