summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorSerhiy Storchaka <storchaka@gmail.com>2022-12-17 12:32:48 (GMT)
committerGitHub <noreply@github.com>2022-12-17 12:32:48 (GMT)
commit0fe61d0838218b18bcff4bf434beba3e8dbcc42b (patch)
tree54d1b62faa69456c80603aeba76953e0f4749a66
parent2b38a9aa747785f6d540e95e6846d6510de6b306 (diff)
downloadcpython-0fe61d0838218b18bcff4bf434beba3e8dbcc42b.zip
cpython-0fe61d0838218b18bcff4bf434beba3e8dbcc42b.tar.gz
cpython-0fe61d0838218b18bcff4bf434beba3e8dbcc42b.tar.bz2
gh-100272: Fix JSON serialization of OrderedDict (GH-100273)
It now preserves the order of keys.
-rw-r--r--Lib/test/test_json/test_default.py11
-rw-r--r--Misc/NEWS.d/next/Library/2022-12-15-18-28-13.gh-issue-100272.D1O9Ey.rst1
-rw-r--r--Modules/_json.c7
3 files changed, 15 insertions, 4 deletions
diff --git a/Lib/test/test_json/test_default.py b/Lib/test/test_json/test_default.py
index 9b8325e..3ce1668 100644
--- a/Lib/test/test_json/test_default.py
+++ b/Lib/test/test_json/test_default.py
@@ -1,3 +1,4 @@
+import collections
from test.test_json import PyTest, CTest
@@ -7,6 +8,16 @@ class TestDefault:
self.dumps(type, default=repr),
self.dumps(repr(type)))
+ def test_ordereddict(self):
+ od = collections.OrderedDict(a=1, b=2, c=3, d=4)
+ od.move_to_end('b')
+ self.assertEqual(
+ self.dumps(od),
+ '{"a": 1, "c": 3, "d": 4, "b": 2}')
+ self.assertEqual(
+ self.dumps(od, sort_keys=True),
+ '{"a": 1, "b": 2, "c": 3, "d": 4}')
+
class TestPyDefault(TestDefault, PyTest): pass
class TestCDefault(TestDefault, CTest): pass
diff --git a/Misc/NEWS.d/next/Library/2022-12-15-18-28-13.gh-issue-100272.D1O9Ey.rst b/Misc/NEWS.d/next/Library/2022-12-15-18-28-13.gh-issue-100272.D1O9Ey.rst
new file mode 100644
index 0000000..2fb0892
--- /dev/null
+++ b/Misc/NEWS.d/next/Library/2022-12-15-18-28-13.gh-issue-100272.D1O9Ey.rst
@@ -0,0 +1 @@
+Fix JSON serialization of OrderedDict. It now preserves the order of keys.
diff --git a/Modules/_json.c b/Modules/_json.c
index 429b4ee..6879ad3 100644
--- a/Modules/_json.c
+++ b/Modules/_json.c
@@ -1570,10 +1570,9 @@ encoder_listencode_dict(PyEncoderObject *s, _PyUnicodeWriter *writer,
*/
}
- if (s->sort_keys) {
-
- items = PyDict_Items(dct);
- if (items == NULL || PyList_Sort(items) < 0)
+ if (s->sort_keys || !PyDict_CheckExact(dct)) {
+ items = PyMapping_Items(dct);
+ if (items == NULL || (s->sort_keys && PyList_Sort(items) < 0))
goto bail;
for (Py_ssize_t i = 0; i < PyList_GET_SIZE(items); i++) {