Commit b26c1716 authored by INADA Naoki's avatar INADA Naoki

Issue #18219: Optimize csv.DictWriter for large number of columns.

Patch by Mariatta Wijaya.
parent 166dbf3e
...@@ -195,10 +195,12 @@ The :mod:`csv` module defines the following classes: ...@@ -195,10 +195,12 @@ The :mod:`csv` module defines the following classes:
written if the dictionary is missing a key in *fieldnames*. If the written if the dictionary is missing a key in *fieldnames*. If the
dictionary passed to the :meth:`writerow` method contains a key not found in dictionary passed to the :meth:`writerow` method contains a key not found in
*fieldnames*, the optional *extrasaction* parameter indicates what action to *fieldnames*, the optional *extrasaction* parameter indicates what action to
take. If it is set to ``'raise'`` a :exc:`ValueError` is raised. If it is take.
set to ``'ignore'``, extra values in the dictionary are ignored. Any other If it is set to ``'raise'``, the default value, a :exc:`ValueError`
optional or keyword arguments are passed to the underlying :class:`writer` is raised.
instance. If it is set to ``'ignore'``, extra values in the dictionary are ignored.
Any other optional or keyword arguments are passed to the underlying
:class:`writer` instance.
Note that unlike the :class:`DictReader` class, the *fieldnames* parameter Note that unlike the :class:`DictReader` class, the *fieldnames* parameter
of the :class:`DictWriter` is not optional. Since Python's :class:`dict` of the :class:`DictWriter` is not optional. Since Python's :class:`dict`
......
...@@ -145,7 +145,7 @@ class DictWriter: ...@@ -145,7 +145,7 @@ class DictWriter:
def _dict_to_list(self, rowdict): def _dict_to_list(self, rowdict):
if self.extrasaction == "raise": if self.extrasaction == "raise":
wrong_fields = [k for k in rowdict if k not in self.fieldnames] wrong_fields = rowdict.keys() - self.fieldnames
if wrong_fields: if wrong_fields:
raise ValueError("dict contains fields not in fieldnames: " raise ValueError("dict contains fields not in fieldnames: "
+ ", ".join([repr(x) for x in wrong_fields])) + ", ".join([repr(x) for x in wrong_fields]))
......
...@@ -626,6 +626,24 @@ class TestDictFields(unittest.TestCase): ...@@ -626,6 +626,24 @@ class TestDictFields(unittest.TestCase):
self.assertNotIn("'f2'", exception) self.assertNotIn("'f2'", exception)
self.assertIn("1", exception) self.assertIn("1", exception)
def test_typo_in_extrasaction_raises_error(self):
fileobj = StringIO()
self.assertRaises(ValueError, csv.DictWriter, fileobj, ['f1', 'f2'],
extrasaction="raised")
def test_write_field_not_in_field_names_raise(self):
fileobj = StringIO()
writer = csv.DictWriter(fileobj, ['f1', 'f2'], extrasaction="raise")
dictrow = {'f0': 0, 'f1': 1, 'f2': 2, 'f3': 3}
self.assertRaises(ValueError, csv.DictWriter.writerow, writer, dictrow)
def test_write_field_not_in_field_names_ignore(self):
fileobj = StringIO()
writer = csv.DictWriter(fileobj, ['f1', 'f2'], extrasaction="ignore")
dictrow = {'f0': 0, 'f1': 1, 'f2': 2, 'f3': 3}
csv.DictWriter.writerow(writer, dictrow)
self.assertEqual(fileobj.getvalue(), "1,2\r\n")
def test_read_dict_fields(self): def test_read_dict_fields(self):
with TemporaryFile("w+") as fileobj: with TemporaryFile("w+") as fileobj:
fileobj.write("1,2,abc\r\n") fileobj.write("1,2,abc\r\n")
......
...@@ -20,6 +20,9 @@ Core and Builtins ...@@ -20,6 +20,9 @@ Core and Builtins
Library Library
------- -------
- Issue #18219: Optimize csv.DictWriter for large number of columns.
Patch by Mariatta Wijaya.
- Issue #28448: Fix C implemented asyncio.Future didn't work on Windows. - Issue #28448: Fix C implemented asyncio.Future didn't work on Windows.
- Issue #28480: Fix error building socket module when multithreading is - Issue #28480: Fix error building socket module when multithreading is
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment