Python 요청을 사용하여 JSON 게시


633

클라이언트에서 서버로 JSON을 POST해야합니다. Python 2.7.1 및 simplejson을 사용하고 있습니다. 클라이언트가 요청을 사용하고 있습니다. 서버는 CherryPy입니다. 서버에서 하드 코딩 된 JSON을 얻을 수 있지만 (코드는 표시되지 않음) JSON을 서버에 POST하려고하면 "400 Bad Request"가 표시됩니다.

내 고객 코드는 다음과 같습니다.

data = {'sender':   'Alice',
    'receiver': 'Bob',
    'message':  'We did it!'}
data_json = simplejson.dumps(data)
payload = {'json_payload': data_json}
r = requests.post("http://localhost:8080", data=payload)

서버 코드는 다음과 같습니다.

class Root(object):

    def __init__(self, content):
        self.content = content
        print self.content  # this works

    exposed = True

    def GET(self):
        cherrypy.response.headers['Content-Type'] = 'application/json'
        return simplejson.dumps(self.content)

    def POST(self):
        self.content = simplejson.loads(cherrypy.request.body.read())

어떤 아이디어?


나는 문서에서 곧바로 예제를 제거했다 .
Charles R

내 의견은 여전히 ​​유효합니다-CherryPy는 인수로 클래스 __init__메소드를 호출하지 않습니다 content(제공 한 링크에서 주장하지 않습니다). 그들이 가지고있는 자세한 예에서, 사용자는 호출하는 코드를 __init__제공하고 인수를 제공합니다. 여기서는 보지 못했습니다. 따라서 # this works주석과 관련이 있을 때 객체의 상태를 알 수 없습니다 .
Nick Bastin

1
인스턴스가 생성 된 라인을 보시겠습니까?
Charles R

예, 테스트하기 위해 예제를 시작하려고했지만 어떻게 인스턴스화했는지 잘 모르겠습니다.
Nick Bastin

코드가 변경되었습니다. 나는 이제 추가적인 논쟁없이 그것을 만들고 있습니다. cherrypy.quickstart(Root(), '/', conf).
Charles R

답변:


1052

Requests 버전 2.4.2부터는 호출에서 'json'매개 변수를 대신 사용할 수 있습니다.

>>> import requests
>>> r = requests.post('http://httpbin.org/post', json={"key": "value"})
>>> r.status_code
200
>>> r.json()
{'args': {},
 'data': '{"key": "value"}',
 'files': {},
 'form': {},
 'headers': {'Accept': '*/*',
             'Accept-Encoding': 'gzip, deflate',
             'Connection': 'close',
             'Content-Length': '16',
             'Content-Type': 'application/json',
             'Host': 'httpbin.org',
             'User-Agent': 'python-requests/2.4.3 CPython/3.4.0',
             'X-Request-Id': 'xx-xx-xx'},
 'json': {'key': 'value'},
 'origin': 'x.x.x.x',
 'url': 'http://httpbin.org/post'}

편집 :이 기능은 공식 문서에 추가되었습니다. 여기에서 볼 수 있습니다 : 문서 요청


114
나는 당신의 대답에 걸려 넘어지기 전에 얼마나 많은 시간을 낭비했는지 믿을 수 없습니다. 요청 문서는 절대적으로 아무것도 없다, 업그레이드 할 필요가 json매개 변수. github.com/kennethreitz/requests/blob/…에
IAmKale

1
2.4.2부터 더 관용적이므로 이것을 허용 된 답변으로 설정하십시오. 미친 유니 코드의 경우 작동하지 않을 수 있습니다.
Charles R

@IAmKale과 같은 신발을 신었습니다. 이것은 AWS의 API 게이트웨이로 인한 어려움을 덜어줍니다. 기본적으로 JSON 형식의 POST 데이터가 필요합니다.
jstudios

1
바보처럼 나는 application / json과 함께 data 매개 변수를 사용하려고 시도했습니다 콘텐츠 유형 :(
Illegal Operator

dict 객체를 가져 와서 보내기 전에 json.dumps (object)를 수행하는 예제를 보았습니다. 이 작업을 수행하지 마십시오 ... JSON이 엉망이됩니다. 위의 내용은 완벽합니다. 파이썬 객체를 전달하면 완벽한 json으로 변합니다.
MydKnight

376

헤더 정보가 누락되었습니다. 다음과 같이 작동합니다.

url = "http://localhost:8080"
data = {'sender': 'Alice', 'receiver': 'Bob', 'message': 'We did it!'}
headers = {'Content-type': 'application/json', 'Accept': 'text/plain'}
r = requests.post(url, data=json.dumps(data), headers=headers)

잘 잡으십시오-나는 당신 application/json을 보았고 GET어떻게 든 당신이 요청에 그것을 제공하지 않았다는 것을 놓쳤습니다. 에서 무언가를 반품하거나을 POST받을 수도 있습니다 500.
Nick Bastin

필요하지 않은 것 같습니다. 나는 인쇄 할 때 r, 나는 얻을 <Response [200]>.
Charles R

서버 측 에서이 JSON을 어떻게 검색합니까?
VaidAbhishek

r = requests.get ( ' localhost : 8080' ) c = r.content 결과 = simplejson.loads (c)
Charles R

1
json.dumps여기에 사용하기 전에 작은 머리 . data의 매개 변수를 requests사전에 작품 잘. 문자열로 변환 할 필요가 없습니다.
Advait S

71

요청 2.4.2 ( https://pypi.python.org/pypi/requests )에서 "json"매개 변수가 지원됩니다. "Content-Type"을 지정할 필요가 없습니다. 따라서 더 짧은 버전 :

requests.post('http://httpbin.org/post', json={'test': 'cheers'})

29

더 좋은 방법은 :

url = "http://xxx.xxxx.xx"

datas = {"cardno":"6248889874650987","systemIdentify":"s08","sourceChannel": 12}

headers = {'Content-type': 'application/json'}

rsp = requests.post(url, json=datas, headers=headers)

18
Content-type: application/json는 AS 중복 json=이미 그 힌트.
Moshe

1
@Moshe는 완전히 동의하지만 최신 버전을 요청하려면 Elasticsearch 서버를 설정해야합니다. Content-type
devesh

@Moshe, 콘텐츠 유형은 무엇입니까 경우 text/html; charset=UTF-8. 그런 다음 위의 작동하지 않습니다?
Anu

2
" 더 나은 방법은 " 정답 후 3 년 동안 잘못된 답변 을 게시하지 않는 것입니다. -1
CONvid19

3

Python 3.5 이상에서 완벽하게 작동

고객:

import requests
data = {'sender':   'Alice',
    'receiver': 'Bob',
    'message':  'We did it!'}
r = requests.post("http://localhost:8080", json={'json_payload': data})

섬기는 사람:

class Root(object):

    def __init__(self, content):
        self.content = content
        print self.content  # this works

    exposed = True

    def GET(self):
        cherrypy.response.headers['Content-Type'] = 'application/json'
        return simplejson.dumps(self.content)

    @cherrypy.tools.json_in()
    @cherrypy.tools.json_out()
    def POST(self):
        self.content = cherrypy.request.json
        return {'status': 'success', 'message': 'updated'}

3

(데이터 / json / 파일) 사이의 어떤 매개 변수를 사용해야합니까? 실제로 ContentType이라는 요청 헤더에 따라 다릅니다 (일반적으로 브라우저의 개발자 도구를 통해 확인),

Content-Type이 application / x-www-form-urlencoded 인 경우 코드는 다음과 같아야합니다.

requests.post(url, data=jsonObj)

Content-Type이 application / json 인 경우 코드는 다음 중 하나 여야합니다.

requests.post(url, json=jsonObj)
requests.post(url, data=jsonstr, headers={"Content-Type":"application/json"})

Content-Type이 multipart / form-data 인 경우 파일을 업로드하는 데 사용되므로 코드는 다음과 같아야합니다.

requests.post(url, files=xxxx)

예수 그리스도, 감사합니다. 나는 몇 분 전에 머리를 내밀었다.
Vahagn Tumanyan

당신을 도울 수있어서 다행
:)
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.