summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMiss Islington (bot) <31488909+miss-islington@users.noreply.github.com>2017-12-12 10:56:40 (GMT)
committerSerhiy Storchaka <storchaka@gmail.com>2017-12-12 10:56:40 (GMT)
commitce5a3cd9b15c9379753aefabd696bff11495cbbb (patch)
tree00853192072af43d64be4aca732f2fca9298c6c4
parent82adaf5ffe425f3d3360e972df03ff15584b4035 (diff)
downloadcpython-ce5a3cd9b15c9379753aefabd696bff11495cbbb.zip
cpython-ce5a3cd9b15c9379753aefabd696bff11495cbbb.tar.gz
cpython-ce5a3cd9b15c9379753aefabd696bff11495cbbb.tar.bz2
bpo-32255: Always quote a single empty field when write into a CSV file. (GH-4769) (#4810)
This allows to distinguish an empty row from a row consisting of a single empty field. (cherry picked from commit 2001900b0c02a397d8cf1d776a7cc7fcb2a463e3)
-rw-r--r--Lib/test/test_csv.py21
-rw-r--r--Misc/ACKS1
-rw-r--r--Misc/NEWS.d/next/Library/2017-12-12-07-29-06.bpo-32255.2bfNmM.rst3
-rw-r--r--Modules/_csv.c2
4 files changed, 25 insertions, 2 deletions
diff --git a/Lib/test/test_csv.py b/Lib/test/test_csv.py
index 03ab184..fe24801 100644
--- a/Lib/test/test_csv.py
+++ b/Lib/test/test_csv.py
@@ -207,10 +207,29 @@ class Test_Csv(unittest.TestCase):
with TemporaryFile("w+", newline='') as fileobj:
writer = csv.writer(fileobj)
self.assertRaises(TypeError, writer.writerows, None)
- writer.writerows([['a','b'],['c','d']])
+ writer.writerows([['a', 'b'], ['c', 'd']])
fileobj.seek(0)
self.assertEqual(fileobj.read(), "a,b\r\nc,d\r\n")
+ def test_writerows_with_none(self):
+ with TemporaryFile("w+", newline='') as fileobj:
+ writer = csv.writer(fileobj)
+ writer.writerows([['a', None], [None, 'd']])
+ fileobj.seek(0)
+ self.assertEqual(fileobj.read(), "a,\r\n,d\r\n")
+
+ with TemporaryFile("w+", newline='') as fileobj:
+ writer = csv.writer(fileobj)
+ writer.writerows([[None], ['a']])
+ fileobj.seek(0)
+ self.assertEqual(fileobj.read(), '""\r\na\r\n')
+
+ with TemporaryFile("w+", newline='') as fileobj:
+ writer = csv.writer(fileobj)
+ writer.writerows([['a'], [None]])
+ fileobj.seek(0)
+ self.assertEqual(fileobj.read(), 'a\r\n""\r\n')
+
@support.cpython_only
def test_writerows_legacy_strings(self):
import _testcapi
diff --git a/Misc/ACKS b/Misc/ACKS
index 5f21c4b..0a0823c 100644
--- a/Misc/ACKS
+++ b/Misc/ACKS
@@ -1525,6 +1525,7 @@ Joel Taddei
Arfrever Frehtes Taifersar Arahesis
Hideaki Takahashi
Takase Arihiro
+Licht Takeuchi
Indra Talip
Neil Tallim
Geoff Talvola
diff --git a/Misc/NEWS.d/next/Library/2017-12-12-07-29-06.bpo-32255.2bfNmM.rst b/Misc/NEWS.d/next/Library/2017-12-12-07-29-06.bpo-32255.2bfNmM.rst
new file mode 100644
index 0000000..dafee67
--- /dev/null
+++ b/Misc/NEWS.d/next/Library/2017-12-12-07-29-06.bpo-32255.2bfNmM.rst
@@ -0,0 +1,3 @@
+A single empty field is now always quoted when written into a CSV file.
+This allows to distinguish an empty row from a row consisting of a single empty field.
+Patch by Licht Takeuchi.
diff --git a/Modules/_csv.c b/Modules/_csv.c
index 7a78541..c61eae5 100644
--- a/Modules/_csv.c
+++ b/Modules/_csv.c
@@ -1245,7 +1245,7 @@ csv_writerow(WriterObj *self, PyObject *seq)
if (PyErr_Occurred())
return NULL;
- if (self->num_fields > 0 && self->rec_size == 0) {
+ if (self->num_fields > 0 && self->rec_len == 0) {
if (dialect->quoting == QUOTE_NONE) {
PyErr_Format(_csvstate_global->error_obj,
"single empty field record must be quoted");