core.csv
Offers tools to deal with csv (and xls, xlsx) files.
Attributes
Classes
Provides access to a csv file. |
Functions
|
Since encoding detection is hard to get right (and work correctly |
|
Takes the given csv string and returns the dialect or raises an error. |
|
Normalizes a header value to be as uniform as possible. |
|
Takes an XLS file and returns a csv file using the given worksheet |
|
Takes an XLS file and returns a csv file using the given worksheet |
|
Takes an XLS/XLSX file and returns a csv file using the given worksheet |
|
|
|
|
Returns all keys of a list of dicts in an ordered tuple. |
|
|
Takes a list of dictionaries and returns a csv. |
|
Takes a list of dictionaries and returns a xlsx. |
Like to |
|
|
Ensuring the title of the xlsx is valid. |
|
Naive check to see whether name already exists. |
|
Removes all chars from beginning up until and including the first "-". |
|
|
|
Returns a list of indexes of duplicates in a list. |
|
Takes the first line of the given csv string and returns the headers. |
|
Takes a list of normalized headers and matches them up against a |
Module Contents
- class core.csv.CSVFile(csvfile: IO[bytes], expected_headers: collections.abc.Collection[str] | None = None, dialect: type[csv.Dialect] | csv.Dialect | str | None = None, encoding: str | None = None, rename_duplicate_column_names: bool = False, rowtype: None = None)[source]
- class core.csv.CSVFile(csvfile: IO[bytes], expected_headers: collections.abc.Collection[str] | None = None, dialect: type[csv.Dialect] | csv.Dialect | str | None = None, encoding: str | None = None, rename_duplicate_column_names: bool = False, *, rowtype: _RowType[_RowT])
Bases:
Generic
[_RowT
]Provides access to a csv file.
- Parameters:
csvfile –
The csv file to be accessed. Must be an open file (not a path), opened in binary mode. For example:
with open(path, 'rb') as f: csv = CSVFile(f)
expected_headers –
The expected headers if known. Expected headers are headers which must exist in the CSV file. There may be additional headers.
If the headers are slightly misspelled, a matching algorithm tries to guess the correct header, without accidentally matching the wrong headers.
See
match_headers()
for more information.If the no expected_headers are passed, no checks are done, but the headers are still available. Headers matching is useful if a user provides the CSV and it might be wrong.
If it is impossible for misspellings to occurr, the expected headers don’t have to be specified.
dialect – The CSV dialect to expect. By default, the dialect will be guessed using Python’s heuristic.
encoding – The CSV encoding to expect. By default, the encoding will be guessed and will either be UTF-8 or CP1252.
rename_duplicate_column_names – It is possible to rename duplicate column names to deal with super wacky files. If this option is set and a duplicate header is found, a suffix is appended to the column name rather than throwing a DuplicateColumnNamesError.
rowtype –
An alternative rowtype for the resulting rows. This should be a callable that receives a rownumber key/value and all the other keys/values found in the csv. The keys are normalized and are valid Python identifiers usable as attribute names.
Defaults to a namedtuple created using the found headers.
Once the csv file is open, the records can be acceessed as follows:
with open(path, 'rb') as f: csv = CSVFile(f) for line in csv.lines: csv.my_field # access the column with the 'my_field' header
- core.csv.detect_encoding(csvfile: IO[bytes]) str [source]
Since encoding detection is hard to get right (and work correctly every time), we limit ourselves here to UTF-8 or CP1252, whichever works first. CP1252 is basically the csv format you get if you use windows and excel and it is a superset of ISO-8859-1/LATIN1.
- core.csv.sniff_dialect(csv: str) type[csv.Dialect] [source]
Takes the given csv string and returns the dialect or raises an error. Works just like Python’s built in sniffer, just that it is a bit more conservative and doesn’t just accept any kind of character as csv delimiter.
- core.csv.normalize_header(header: str) str [source]
Normalizes a header value to be as uniform as possible.
- This includes:
stripping the whitespace around it
lowercasing everything
transliterating unicode (e.g. ‘ä’ becomes ‘a’)
removing duplicate whitespace inside it
- core.csv.convert_xlsx_to_csv(xlsx: IO[bytes], sheet_name: str | None = None) io.BytesIO [source]
Takes an XLS file and returns a csv file using the given worksheet name or the first worksheet found.
- core.csv.convert_xls_to_csv(xls: IO[bytes], sheet_name: str | None = None) io.BytesIO [source]
Takes an XLS file and returns a csv file using the given worksheet name or the first worksheet found.
- core.csv.convert_excel_to_csv(file: IO[bytes], sheet_name: str | None = None) io.BytesIO [source]
Takes an XLS/XLSX file and returns a csv file using the given worksheet name or the first worksheet found.
- core.csv.get_keys_from_list_of_dicts(rows: collections.abc.Iterable[dict[_SupportsRichComparisonT, Any]], key: None = None, reverse: bool = False) tuple[_SupportsRichComparisonT, Ellipsis] [source]
- core.csv.get_keys_from_list_of_dicts(rows: collections.abc.Iterable[dict[_T, Any]], key: KeyFunc[_T], reverse: bool = False) tuple[_T, Ellipsis]
Returns all keys of a list of dicts in an ordered tuple.
If the list of dicts is irregular, the keys found in later rows are added at the end of the list.
Note that the order of keys is otherwise defined by the order of the keys of the dictionaries. So if ordered dictionaries are used, the order is defined. If regular dictionaries are used, the order is undefined.
Alternatively, a key and a reverse flag may be provided which will be used to order the fields. If the list of fields is specified, the key and the reverse flag is ignored.
- core.csv.convert_list_of_dicts_to_csv(rows: collections.abc.Iterable[dict[str, Any]], fields: collections.abc.Sequence[str] | None = None, key: KeyFunc[str] | None = None, reverse: bool = False) str [source]
Takes a list of dictionaries and returns a csv.
If no fields are provided, all fields are included in the order of the keys of the first dict. With regular dictionaries this is random. Use an ordered dict or provide a list of fields to have a fixed order.
Alternatively, a key and a reverse flag may be provided which will be used to order the fields. If the list of fields is specified, the key and the reverse flag is ignored.
The function returns a string created in memory. Therefore this function is limited to small-ish datasets.
- core.csv.convert_list_of_dicts_to_xlsx(rows: collections.abc.Iterable[dict[str, Any]], fields: collections.abc.Sequence[str] | None = None, key: KeyFunc[str] | None = None, reverse: bool = False) bytes [source]
Takes a list of dictionaries and returns a xlsx.
This behaves the same way as
convert_list_of_dicts_to_csv()
.
- core.csv.convert_list_of_list_of_dicts_to_xlsx(row_list: collections.abc.Sequence[collections.abc.Iterable[dict[str, Any]]], titles_list: collections.abc.Sequence[str], key_list: collections.abc.Sequence[KeyFunc[str] | None] | None = None, reverse: bool = False) bytes [source]
Like to
convert_list_of_dicts_to_xlsx()
, but operates on a list instead of in a single item.
- core.csv.normalize_sheet_titles(titles: collections.abc.Sequence[str]) list[str] [source]
Ensuring the title of the xlsx is valid.
- core.csv.avoid_duplicate_name(titles: collections.abc.Sequence[str], title: str) str [source]
Naive check to see whether name already exists. If name does exist suggest a name using an incrementer Duplicates are case-insensitive
- core.csv.remove_first_word(title: str) str [source]
Removes all chars from beginning up until and including the first “-“.
- core.csv.list_duplicates_index(a: collections.abc.Sequence[Any]) list[int] [source]
Returns a list of indexes of duplicates in a list. for example:
a = [1, 2, 3, 2, 1, 5, 6, 5, 5, 5] list_duplicates_index(a) == [3, 4, 7, 8, 9]
- core.csv.parse_header(csv: str, dialect: type[csv.Dialect] | csv.Dialect | str | None = None, rename_duplicate_column_names: bool = False) list[str] [source]
Takes the first line of the given csv string and returns the headers.
Headers are normalized (stripped and normalized) and expected to be unique. The dialect is sniffed, if not provided.
- Returns:
A list of headers in the order of appearance.
- core.csv.match_headers(headers: collections.abc.Collection[str], expected: collections.abc.Collection[str]) list[str] [source]
Takes a list of normalized headers and matches them up against a list of expected headers.
The headers may differ from the expected headers. This function tries to match them up using the Levenshtein distance. It does so somewhat carefully by calculating a sane distance using the input.
The passed headers are expected to be have been normalized by
normalize_header()
, since we usually will pass the result ofparse_header()
to this function.For example:
match_headers( headers=['firstname', 'lastname'], expected=['last_name', 'first_name'] )
Results in:
['first_name', 'last_name']
If no match is possible, an
MissingColumnsError
, orAmbiguousColumnsError
orDuplicateColumnNamesError
error is raised.- Returns:
The matched headers in the order of appearance.