add metavar option and rework storage logic
This commit is contained in:
@@ -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")
|
||||||
|
|||||||
@@ -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)
|
||||||
|
|||||||
Reference in New Issue
Block a user