Kaydet (Commit) af8fcfae authored tarafından Skip Montanaro's avatar Skip Montanaro

Merged revisions 65605 via svnmerge from

svn+ssh://pythondev@svn.python.org/python/trunk

........
  r65605 | skip.montanaro | 2008-08-08 17:52:51 -0500 (Fri, 08 Aug 2008) | 1 line

  accept issue 3436
........
üst bea655e5
...@@ -372,6 +372,18 @@ Reader objects have the following public attributes: ...@@ -372,6 +372,18 @@ Reader objects have the following public attributes:
DictReader objects have the following public attribute:
.. attribute:: csvreader.fieldnames
If not passed as a parameter when creating the object, this attribute is
initialized upon first access or when the first record is read from the
file.
.. versionchanged:: 2.6
Writer Objects Writer Objects
-------------- --------------
......
...@@ -68,7 +68,7 @@ register_dialect("excel-tab", excel_tab) ...@@ -68,7 +68,7 @@ register_dialect("excel-tab", excel_tab)
class DictReader: class DictReader:
def __init__(self, f, fieldnames=None, restkey=None, restval=None, def __init__(self, f, fieldnames=None, restkey=None, restval=None,
dialect="excel", *args, **kwds): dialect="excel", *args, **kwds):
self.fieldnames = fieldnames # list of keys for the dict self._fieldnames = fieldnames # list of keys for the dict
self.restkey = restkey # key to catch long rows self.restkey = restkey # key to catch long rows
self.restval = restval # default value for short rows self.restval = restval # default value for short rows
self.reader = reader(f, dialect, *args, **kwds) self.reader = reader(f, dialect, *args, **kwds)
...@@ -78,10 +78,24 @@ class DictReader: ...@@ -78,10 +78,24 @@ class DictReader:
def __iter__(self): def __iter__(self):
return self return self
@property
def fieldnames(self):
if self._fieldnames is None:
try:
self._fieldnames = next(self.reader)
except StopIteration:
pass
self.line_num = self.reader.line_num
return self._fieldnames
@fieldnames.setter
def fieldnames(self, value):
self._fieldnames = value
def __next__(self): def __next__(self):
row = next(self.reader) if self.line_num == 0:
if self.fieldnames is None: # Used only for its side effect.
self.fieldnames = row self.fieldnames
row = next(self.reader) row = next(self.reader)
self.line_num = self.reader.line_num self.line_num = self.reader.line_num
......
...@@ -544,6 +544,29 @@ class TestDictFields(unittest.TestCase): ...@@ -544,6 +544,29 @@ class TestDictFields(unittest.TestCase):
fileobj.seek(0) fileobj.seek(0)
reader = csv.DictReader(fileobj) reader = csv.DictReader(fileobj)
self.assertEqual(next(reader), {"f1": '1', "f2": '2', "f3": 'abc'}) self.assertEqual(next(reader), {"f1": '1', "f2": '2', "f3": 'abc'})
self.assertEqual(reader.fieldnames, ["f1", "f2", "f3"])
# Two test cases to make sure existing ways of implicitly setting
# fieldnames continue to work. Both arise from discussion in issue3436.
def test_read_dict_fieldnames_from_file(self):
with TemporaryFile("w+") as fileobj:
fileobj.write("f1,f2,f3\r\n1,2,abc\r\n")
fileobj.seek(0)
reader = csv.DictReader(fileobj,
fieldnames=next(csv.reader(fileobj)))
self.assertEqual(reader.fieldnames, ["f1", "f2", "f3"])
self.assertEqual(next(reader), {"f1": '1', "f2": '2', "f3": 'abc'})
def test_read_dict_fieldnames_chain(self):
import itertools
with TemporaryFile("w+") as fileobj:
fileobj.write("f1,f2,f3\r\n1,2,abc\r\n")
fileobj.seek(0)
reader = csv.DictReader(fileobj)
first = next(reader)
for row in itertools.chain([first], reader):
self.assertEqual(reader.fieldnames, ["f1", "f2", "f3"])
self.assertEqual(row, {"f1": '1', "f2": '2', "f3": 'abc'})
def test_read_long(self): def test_read_long(self):
with TemporaryFile("w+") as fileobj: with TemporaryFile("w+") as fileobj:
...@@ -568,6 +591,7 @@ class TestDictFields(unittest.TestCase): ...@@ -568,6 +591,7 @@ class TestDictFields(unittest.TestCase):
fileobj.write("f1,f2\r\n1,2,abc,4,5,6\r\n") fileobj.write("f1,f2\r\n1,2,abc,4,5,6\r\n")
fileobj.seek(0) fileobj.seek(0)
reader = csv.DictReader(fileobj, restkey="_rest") reader = csv.DictReader(fileobj, restkey="_rest")
self.assertEqual(reader.fieldnames, ["f1", "f2"])
self.assertEqual(next(reader), {"f1": '1', "f2": '2', self.assertEqual(next(reader), {"f1": '1', "f2": '2',
"_rest": ["abc", "4", "5", "6"]}) "_rest": ["abc", "4", "5", "6"]})
......
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