python-benedict/benedict/serializers/csv.py

70 lines
2.1 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
2020-01-20 16:03:59 +00:00
from __future__ import absolute_import
from benedict.serializers.abstract import AbstractSerializer
from benedict.utils import type_util
from six import StringIO
import csv
class CSVSerializer(AbstractSerializer):
2022-02-13 10:56:44 +00:00
"""
This class describes a csv serializer.
"""
def __init__(self):
super(CSVSerializer, self).__init__()
def decode(self, s, **kwargs):
# kwargs.setdefault('delimiter', ',')
2022-02-13 10:35:43 +00:00
if kwargs.pop("quote", False):
# TODO: add tests coverage
2022-02-13 10:35:43 +00:00
kwargs.setdefault("quoting", csv.QUOTE_ALL)
columns = kwargs.pop("columns", None)
columns_row = kwargs.pop("columns_row", True)
f = StringIO(s)
r = csv.reader(f, **kwargs)
ln = 0
data = []
for row in r:
if ln == 0 and columns_row:
if not columns:
columns = row
ln += 1
continue
d = dict(zip(columns, row))
data.append(d)
ln += 1
return data
def encode(self, d, **kwargs):
l = d
# kwargs.setdefault('delimiter', ',')
2022-02-13 10:35:43 +00:00
if kwargs.pop("quote", False):
kwargs.setdefault("quoting", csv.QUOTE_ALL)
kwargs.setdefault("lineterminator", "\n")
columns = kwargs.pop("columns", None)
columns_row = kwargs.pop("columns_row", True)
if not columns and len(l) and type_util.is_dict(l[0]):
keys = [str(key) for key in l[0].keys()]
columns = list(sorted(keys))
f = StringIO()
w = csv.writer(f, **kwargs)
if columns_row and columns:
w.writerow(columns)
for item in l:
if type_util.is_dict(item):
2022-02-13 10:35:43 +00:00
row = [item.get(key, "") for key in columns]
elif type_util.is_collection(item):
# TODO: add tests coverage
row = item
else:
# TODO: add tests coverage
row = [item]
w.writerow(row)
data = f.getvalue()
return data