Difference between data and json parameters in python requests package Difference between data and json parameters in python requests package json json

Difference between data and json parameters in python requests package


To answer my own question, it appears my two examples above do the same thing and that using the json parameter does indeed set the content-type in the headers to application/json. In my first example above using the data parameter, the content-type in the headers would need to be set manually.


As of 2021 I feel that the requests documentation is more clear about the difference, but I still have created a PR to make it more clear.


PS This does NOT answer the OP question but if the FIRST code would be a bit different:

import requestsimport jsond = {'a': 1}response = requests.post(url, data=d)

- note that the dict d is NOT converted to JSON string here!

And if the second code would be the same (copying it for completeness):

import requestsimport jsond = {'a': 1}response = requests.post(url, json=d)

...then the result would be quite a lot different.

First code would generate a request with content type set to application/x-www-form-urlencoded and the data in this format, so: "a=1"

The second code would generate a request with content type set to application/json and in fact the data in this format, so {"a": 1} - a JSON string.


Talking only from my experience here, but please note that it should be preferred to use the json field with the dict, rather than dumping the dict in the data field.

Again, talking only from experience, I did not study the code itself, but it seems that the requests library does some more clever json serialization than just json.dumps. When using json.dumps in data field, I have encountered several instances where this resulted in an "value is not a valid dict" error response from the (FastAPI) server. Using the json field instead fixed these issues.

EDIT: I went through the code today. If you use the json parameter, it seems the requests library really only sets Content-Type and dumps it:

from .compat import json as complexjsoncontent_type = 'application/json'body = complexjson.dumps(json)if not isinstance(body, bytes):    body = body.encode('utf-8')

where in the requests.compat, the json is just:

try:    import simplejson as jsonexcept ImportError:    import json

... therefore I really cannot figure out why doing this manually using the data parameter just sometimes fails to work. ¯\_(ツ)_/¯