PyMISP/pymisp/tools/csvloader.py

64 lines
2.5 KiB
Python
Raw Normal View History

2019-04-03 16:28:26 +02:00
#!/usr/bin/env python3
from __future__ import annotations
2019-04-03 16:28:26 +02:00
from pathlib import Path
2019-04-03 16:28:26 +02:00
import csv
from pymisp import MISPObject
class CSVLoader():
2024-01-31 15:20:31 +01:00
def __init__(self, template_name: str, csv_path: Path,
fieldnames: list[str] | None = None, has_fieldnames: bool=False,
delimiter: str = ',', quotechar: str = '"') -> None:
2019-04-03 16:28:26 +02:00
self.template_name = template_name
self.delimiter = delimiter
self.quotechar = quotechar
2019-04-03 16:28:26 +02:00
self.csv_path = csv_path
self.fieldnames = []
if fieldnames:
self.fieldnames = [f.strip() for f in fieldnames]
2019-04-03 16:28:26 +02:00
if not self.fieldnames:
# If the user doesn't pass fieldnames, they must be in the CSV.
2019-04-03 16:28:26 +02:00
self.has_fieldnames = True
else:
self.has_fieldnames = has_fieldnames
2024-01-31 15:20:31 +01:00
def load(self) -> list[MISPObject]:
2019-04-03 16:28:26 +02:00
objects = []
with open(self.csv_path, newline='') as csvfile:
reader = csv.reader(csvfile, delimiter=self.delimiter, quotechar=self.quotechar)
2019-04-03 16:28:26 +02:00
if self.has_fieldnames:
# The file has fieldnames, we either ignore it, or use them as object-relation
fieldnames = [f.strip() for f in reader.__next__()]
2019-04-03 16:28:26 +02:00
if not self.fieldnames:
self.fieldnames = fieldnames
if not self.fieldnames:
2020-05-12 11:34:38 +02:00
raise Exception('No fieldnames, impossible to create objects.')
# Check if the CSV file has a header, and if it matches with the object template
tmp_object = MISPObject(self.template_name)
2024-01-31 15:20:31 +01:00
if not tmp_object._definition or not tmp_object._definition['attributes']:
raise Exception(f'Unable to find the object template ({self.template_name}), impossible to create objects.')
allowed_fieldnames = list(tmp_object._definition['attributes'].keys())
for fieldname in self.fieldnames:
if fieldname not in allowed_fieldnames:
raise Exception(f'{fieldname} is not a valid object relation for {self.template_name}: {allowed_fieldnames}')
2019-04-03 16:28:26 +02:00
for row in reader:
tmp_object = MISPObject(self.template_name)
has_attribute = False
2019-04-03 16:28:26 +02:00
for object_relation, value in zip(self.fieldnames, row):
if value:
has_attribute = True
tmp_object.add_attribute(object_relation, value=value)
if has_attribute:
objects.append(tmp_object)
2019-04-03 16:28:26 +02:00
return objects