Як надіслати запит на пошту як JSON?


105
data = {
        'ids': [12, 3, 4, 5, 6 , ...]
    }
    urllib2.urlopen("http://abc.com/api/posts/create",urllib.urlencode(data))

Я хочу надіслати запит POST, але в одному з полів повинен бути список номерів. Як я можу це зробити? (JSON?)


1
Хіба це вже не список номерів?
Waynn Lue

На це не можна відповісти, не знаючи, який вхід очікує API.
Ніклас Б.

1
@WaynnLue сервер API отримує це як рядок, а не список.
TIMEX

1
Чи потрібно встановлювати заголовки як "application / json" чи щось таке?
TIMEX

Відповіді:


154

Якщо ваш сервер очікує, що POST-запит буде json, вам потрібно буде додати заголовок, а також серіалізувати дані для вашого запиту ...

Python 2.x

import json
import urllib2

data = {
        'ids': [12, 3, 4, 5, 6]
}

req = urllib2.Request('http://example.com/api/posts/create')
req.add_header('Content-Type', 'application/json')

response = urllib2.urlopen(req, json.dumps(data))

Python 3.x

https://stackoverflow.com/a/26876308/496445


Якщо не вказати заголовок, він буде типовим application/x-www-form-urlencodedтипом.


У мене є питання. чи можна додати декілька елементів у заголовок ... як тип вмісту та ідентифікатор клієнта ... @jdi
Омар Джандалі

@OmarJandali, просто зателефонуйте add_header()ще раз для кожного заголовка, який ви хочете додати.
jdi

У мене є таке кодування, але воно нічого не друкує. передбачалося надрукувати URL-адреси та заголовки, але нічого не було надруковано ... req = urllib.Request('http://uat-api.synapsefi.com') req.add_header('X-SP-GATEWAY', 'client_id_asdfeavea561va9685e1gre5ara|client_secret_4651av5sa1edgvawegv1a6we1v5a6s51gv') req.add_header('X-SP-USER-IP', '127.0.0.1') req.add_header('X-SP-USER', '| ge85a41v8e16v1a618gea164g65') req.add_header('Content-Type', 'application/json') print(req)...
Омар Джандалі

urllib2 не було розпізнано, тому я просто використав urllib. Я також отримую помилку із запитом. The view tab.views.profileSetup didn't return an HttpResponse object. It returned None instead. @jdi
Омар Джандалі

@OmarJandali, майте на увазі, що ця відповідь спочатку була надана у 2012 році під Python 2.x. Ви використовуєте Python3, тому імпорт буде різним. Це було б import urllib.requestі зараз urllib.request.Request(). Крім того, друк об'єкта req нічого цікавого не робить. Ви чітко бачите, що заголовки додані друком req.headers. Крім того, я не переконаний, чому це не працює у вашій програмі.
jdi

113

Я рекомендую використовувати неймовірний requestsмодуль.

http://docs.python-requests.org/en/v0.10.7/user/quickstart/#custom-headers

url = 'https://api.github.com/some/endpoint'
payload = {'some': 'data'}
headers = {'content-type': 'application/json'}

response = requests.post(url, data=json.dumps(payload), headers=headers)

Це дає меніTypeError: post() takes from 1 to 2 positional arguments but 3 were given
zakdances

66

для python 3.4.2 я виявив, що буде працювати:

import urllib.request
import json      

body = {'ids': [12, 14, 50]}  

myurl = "http://www.testmycode.com"
req = urllib.request.Request(myurl)
req.add_header('Content-Type', 'application/json; charset=utf-8')
jsondata = json.dumps(body)
jsondataasbytes = jsondata.encode('utf-8')   # needs to be bytes
req.add_header('Content-Length', len(jsondataasbytes))
print (jsondataasbytes)
response = urllib.request.urlopen(req, jsondataasbytes)

1
Python3.6.2 це спрацювало. Лише додавання заголовка з req.add_header (...) працювало для мене.
Шалін Л.К.

18

Це ідеально підходить для Python 3.5 , якщо URL-адреса містить значення рядка запиту / параметра,

URL-адреса запиту = https://bah2.com/ws/rest/v1/concept/
Значення параметра = 21f6bb43-98a1-419d-8f0c-8133669e40ca

import requests

url = 'https://bahbah2.com/ws/rest/v1/concept/21f6bb43-98a1-419d-8f0c-8133669e40ca'
data = {"name": "Value"}
r = requests.post(url, auth=('username', 'password'), verify=False, json=data)
print(r.status_code)

7
у вашому снайперському коді змінна заголовків залишається невикористаною
shookees

4

Вам потрібно додати заголовок, інакше ви отримаєте помилку http 400. Код добре працює на python2.6, centos5.4

код:

    import urllib2,json

    url = 'http://www.google.com/someservice'
    postdata = {'key':'value'}

    req = urllib2.Request(url)
    req.add_header('Content-Type','application/json')
    data = json.dumps(postdata)

    response = urllib2.urlopen(req,data)

2

Ось приклад того, як використовувати об’єкт urllib.request зі стандартної бібліотеки Python.

import urllib.request
import json
from pprint import pprint

url = "https://app.close.com/hackwithus/3d63efa04a08a9e0/"

values = {
    "first_name": "Vlad",
    "last_name": "Bezden",
    "urls": [
        "https://twitter.com/VladBezden",
        "https://github.com/vlad-bezden",
    ],
}


headers = {
    "Content-Type": "application/json",
    "Accept": "application/json",
}

data = json.dumps(values).encode("utf-8")
pprint(data)

try:
    req = urllib.request.Request(url, data, headers)
    with urllib.request.urlopen(req) as f:
        res = f.read()
    pprint(res.decode())
except Exception as e:
    pprint(e)

1

У пакеті останніх запитів ви можете використовувати jsonпараметр у requests.post()методі для надсилання дикта json, і Content-Typeв заголовку in буде встановлено значення application/json. Не потрібно чітко вказувати заголовок.

import requests

payload = {'key': 'value'}

requests.post(url, json=payload)

Зауважте, що це призведе до POSTed json з одинарними котируваннями, що технічно є недійсним.
Jethro

@Jethro Ви помітили помилки під час використання одиничних лапок? Дійсно використовувати одиничні лапки в Python. Особисто я не зустрічав жодних питань щодо цього.
jdhao

Вибачте, я помилився, я подумав, що мій сервер отримує JSON з однією котировкою, але це виявилося окремою проблемою та деякою оманливою налагодженням. Привіт, це набагато охайніше, ніж потрібно вказувати заголовок вручну!
Jethro

0

Цей мені добре працює з апісом

import requests

data={'Id':id ,'name': name}
r = requests.post( url = 'https://apiurllink', data = data)
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.