add metavar option and rework storage logic

This commit is contained in:
2021-09-10 11:13:36 +02:00
parent 4725dc9784
commit e632f0d511
2 changed files with 52 additions and 38 deletions

View File

@@ -180,6 +180,9 @@ class ActionConfig(BaseConfig):
if "skip_metadata" in cfg: if "skip_metadata" in cfg:
self.add_bool_arg(cfg, "skip_metadata") self.add_bool_arg(cfg, "skip_metadata")
if "metavar" in cfg:
self.add_string_arg(cfg, "metavar")
else: else:
raise RuntimeError( raise RuntimeError(
f"{self['name']}: storage_type: invalid storage type") f"{self['name']}: storage_type: invalid storage type")

View File

@@ -54,26 +54,32 @@ class BaseMailStorage(object):
class FileMailStorage(BaseMailStorage): class FileMailStorage(BaseMailStorage):
"Storage class to store mails on filesystem." "Storage class to store mails on filesystem."
def __init__(self, directory, original=False, skip_metadata=False): def __init__(self, directory, original=False, skip_metadata=False,
metavar=None):
super().__init__() super().__init__()
self.directory = directory self.directory = directory
self.original = original self.original = original
self.skip_metadata = skip_metadata self.skip_metadata = skip_metadata
self.metavar = metavar
self._metadata_suffix = ".metadata" self._metadata_suffix = ".metadata"
def _save_datafile(self, storage_id, data): def get_storageid(self, qid):
timestamp = datetime.now().strftime("%Y%m%d%H%M%S")
return f"{timestamp}_{qid}"
def _get_file_paths(self, storage_id):
datafile = os.path.join(self.directory, storage_id) datafile = os.path.join(self.directory, storage_id)
metafile = f"{datafile}${self._metadata_suffix}"
return metafile, datafile
def _save_datafile(self, datafile, data):
try: try:
with open(datafile, "wb") as f: with open(datafile, "wb") as f:
f.write(data) f.write(data)
except IOError as e: except IOError as e:
raise RuntimeError(f"unable save data file: {e}") raise RuntimeError(f"unable save data file: {e}")
return datafile def _save_metafile(self, metafile, metadata):
def _save_metafile(self, storage_id, metadata):
metafile = os.path.join(
self.directory, f"{storage_id}{self._metadata_suffix}")
try: try:
with open(metafile, "w") as f: with open(metafile, "w") as f:
json.dump(metadata, f, indent=2) json.dump(metadata, f, indent=2)
@@ -81,27 +87,23 @@ class FileMailStorage(BaseMailStorage):
raise RuntimeError(f"unable to save metadata file: {e}") raise RuntimeError(f"unable to save metadata file: {e}")
def _remove(self, storage_id): def _remove(self, storage_id):
datafile = os.path.join(self.directory, storage_id) metafile, datafile = self._get_file_paths(storage_id)
metafile = f"{datafile}{self._metadata_suffix}"
try: try:
os.remove(metafile) os.remove(metafile)
except IOError as e:
raise RuntimeError(f"unable to remove metadata file: {e}")
try:
os.remove(datafile) os.remove(datafile)
except IOError as e: except IOError as e:
raise RuntimeError(f"unable to remove data file: {e}") raise RuntimeError(f"unable to remove file: {e}")
def add(self, data, qid, mailfrom="", recipients=[], subject=""): def add(self, data, qid, mailfrom="", recipients=[], subject=""):
"Add email to file storage and return storage id." "Add email to file storage and return storage id."
super().add(data, qid, mailfrom, recipients) super().add(data, qid, mailfrom, recipients)
timestamp = datetime.now().strftime("%Y%m%d%H%M%S")
storage_id = f"{timestamp}_{qid}" storage_id = self.get_storageid(qid)
metafile, datafile = self._get_file_paths(storage_id)
# save mail # save mail
datafile = self._save_datafile(storage_id, data) self._save_datafile(datafile, data)
if not self.skip_metadata: if not self.skip_metadata:
# save metadata # save metadata
@@ -113,12 +115,12 @@ class FileMailStorage(BaseMailStorage):
"queue_id": qid} "queue_id": qid}
try: try:
self._save_metafile(storage_id, metadata) self._save_metafile(metafile, metadata)
except RuntimeError as e: except RuntimeError as e:
os.remove(datafile) os.remove(datafile)
raise e raise e
return (storage_id, datafile) return storage_id, metafile, datafile
def execute(self, milter, pretend=False, def execute(self, milter, pretend=False,
logger=logging.getLogger(__name__)): logger=logging.getLogger(__name__)):
@@ -135,18 +137,23 @@ class FileMailStorage(BaseMailStorage):
subject = milter.msg["subject"] or "" subject = milter.msg["subject"] or ""
if not pretend: if not pretend:
storage_id, datafile = self.add( storage_id, metafile, datafile = self.add(
data(), milter.qid, mailfrom, recipients, subject) data(), milter.qid, mailfrom, recipients, subject)
logger.info(f"stored message in file {datafile}") logger.info(f"stored message in file {datafile}")
milter.msginfo["vars"]["STORAGEID"] = storage_id else:
milter.msginfo["vars"]["DATAFILE"] = datafile storage_id = self.get_storageid(milter.qid)
metafile, datafile = self._get_file_paths(storage_id)
if self.metavar:
milter.msginfo["vars"][f"{self.metavar}_ID"] = storage_id
milter.msginfo["vars"][f"{self.metavar}_METAFILE"] = metafile
milter.msginfo["vars"][f"{self.metavar}_DATAFILE"] = datafile
def get_metadata(self, storage_id): def get_metadata(self, storage_id):
"Return metadata of email in storage." "Return metadata of email in storage."
super(FileMailStorage, self).get_metadata(storage_id) super(FileMailStorage, self).get_metadata(storage_id)
metafile = os.path.join( metafile, _ = self._get_file_paths(storage_id)
self.directory, f"{storage_id}{self._metadata_suffix}")
if not os.path.isfile(metafile): if not os.path.isfile(metafile):
raise RuntimeError( raise RuntimeError(
f"invalid storage id '{storage_id}'") f"invalid storage id '{storage_id}'")
@@ -204,16 +211,20 @@ class FileMailStorage(BaseMailStorage):
"Delete email from storage." "Delete email from storage."
super(FileMailStorage, self).delete(storage_id, recipients) super(FileMailStorage, self).delete(storage_id, recipients)
if not recipients:
self._remove(storage_id)
return
try: try:
metadata = self.get_metadata(storage_id) metadata = self.get_metadata(storage_id)
except RuntimeError as e: except RuntimeError as e:
raise RuntimeError(f"unable to delete email: {e}") raise RuntimeError(f"unable to delete email: {e}")
if not recipients: metafile, _ = self._get_file_paths(storage_id)
self._remove(storage_id)
else:
if type(recipients) == str: if type(recipients) == str:
recipients = [recipients] recipients = [recipients]
for recipient in recipients: for recipient in recipients:
if recipient not in metadata["recipients"]: if recipient not in metadata["recipients"]:
raise RuntimeError(f"invalid recipient '{recipient}'") raise RuntimeError(f"invalid recipient '{recipient}'")
@@ -221,7 +232,7 @@ class FileMailStorage(BaseMailStorage):
if not metadata["recipients"]: if not metadata["recipients"]:
self._remove(storage_id) self._remove(storage_id)
else: else:
self._save_metafile(storage_id, metadata) self._save_metafile(metafile, metadata)
def get_mail(self, storage_id): def get_mail(self, storage_id):
super(FileMailStorage, self).get_mail(storage_id) super(FileMailStorage, self).get_mail(storage_id)