importers._csv: Refactor header reading into CSVImporterBase.
This commit is contained in:
parent
1b1e2d038c
commit
0f4f83e079
2 changed files with 34 additions and 37 deletions
|
@ -20,15 +20,32 @@ class CSVImporterBase:
|
||||||
ENTRY_SEED = {}
|
ENTRY_SEED = {}
|
||||||
COPIED_FIELDS = {}
|
COPIED_FIELDS = {}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def _read_header_row(cls, row):
|
||||||
|
return {} if len(row) < cls._HEADER_MAX_LEN else None
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def _read_header(cls, input_file):
|
||||||
|
cls._NEEDED_KEYS = cls.NEEDED_FIELDS.union(cls.COPIED_FIELDS)
|
||||||
|
cls._HEADER_MAX_LEN = len(cls._NEEDED_KEYS)
|
||||||
|
header = {}
|
||||||
|
row = None
|
||||||
|
for row in csv.reader(input_file):
|
||||||
|
row_data = cls._read_header_row(row)
|
||||||
|
if row_data is None:
|
||||||
|
break
|
||||||
|
else:
|
||||||
|
header.update(row_data)
|
||||||
|
return header, row
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def can_import(cls, input_file):
|
def can_import(cls, input_file):
|
||||||
in_csv = csv.reader(input_file)
|
_, fields = cls._read_header(input_file)
|
||||||
fields = next(iter(in_csv), [])
|
return cls._NEEDED_KEYS.issubset(fields or ())
|
||||||
return cls.NEEDED_FIELDS.union(cls.COPIED_FIELDS).issubset(fields)
|
|
||||||
|
|
||||||
def __init__(self, input_file):
|
def __init__(self, input_file):
|
||||||
self.in_csv = csv.DictReader(input_file)
|
self.entry_seed, fields = self._read_header(input_file)
|
||||||
self.entry_seed = {}
|
self.in_csv = csv.DictReader(input_file, fields)
|
||||||
|
|
||||||
def __iter__(self):
|
def __iter__(self):
|
||||||
for row in self.in_csv:
|
for row in self.in_csv:
|
||||||
|
|
|
@ -1,9 +1,12 @@
|
||||||
import csv
|
|
||||||
|
|
||||||
from . import _csv
|
from . import _csv
|
||||||
from .. import strparse
|
from .. import strparse
|
||||||
|
|
||||||
class PaymentImporter(_csv.CSVImporterBase):
|
class PaymentImporter(_csv.CSVImporterBase):
|
||||||
|
HEADER_FIELDS = {
|
||||||
|
'Currency': 'currency',
|
||||||
|
'Disbursement ID': 'disbursement_id',
|
||||||
|
'Reference': 'reference',
|
||||||
|
}
|
||||||
DATE_FIELD = 'Date of Donation'
|
DATE_FIELD = 'Date of Donation'
|
||||||
NAME_FIELDS = ['Donor First Name', 'Donor Last Name']
|
NAME_FIELDS = ['Donor First Name', 'Donor Last Name']
|
||||||
DECIMAL_FIELDS = {
|
DECIMAL_FIELDS = {
|
||||||
|
@ -28,37 +31,14 @@ class PaymentImporter(_csv.CSVImporterBase):
|
||||||
NOT_SHARED = 'Not shared by donor'
|
NOT_SHARED = 'Not shared by donor'
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def _read_header(cls, source):
|
def _read_header_row(cls, row):
|
||||||
needed_keys = cls.NEEDED_FIELDS.union(cls.COPIED_FIELDS)
|
row_len = len(row)
|
||||||
header = {}
|
if row_len > 2:
|
||||||
for row in csv.reader(source):
|
return None
|
||||||
row_len = len(row)
|
elif row_len == 2 and row[0] in cls.HEADER_FIELDS:
|
||||||
if row_len < 2:
|
return {cls.HEADER_FIELDS[row[0]]: row[1]}
|
||||||
pass
|
|
||||||
elif row_len == 2:
|
|
||||||
header[row[0]] = row[1]
|
|
||||||
elif needed_keys.issubset(row):
|
|
||||||
return header, csv.DictReader(source, row)
|
|
||||||
else:
|
|
||||||
break
|
|
||||||
raise ValueError("source is not a Benevity CSV")
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def can_import(cls, input_file):
|
|
||||||
try:
|
|
||||||
header, _ = cls._read_header(input_file)
|
|
||||||
except ValueError:
|
|
||||||
return False
|
|
||||||
else:
|
else:
|
||||||
return bool(header)
|
return {}
|
||||||
|
|
||||||
def __init__(self, input_file):
|
|
||||||
header, self.in_csv = self._read_header(input_file)
|
|
||||||
self.entry_seed = {
|
|
||||||
'currency': header['Currency'],
|
|
||||||
'disbursement_id': header['Disbursement ID'],
|
|
||||||
'reference': header['Payment Reference'],
|
|
||||||
}
|
|
||||||
|
|
||||||
def _read_row(self, row):
|
def _read_row(self, row):
|
||||||
try:
|
try:
|
||||||
|
|
Loading…
Reference in a new issue