From 4903e4290d2e0eba0d7ba551cb42f6b5051e7ec3 Mon Sep 17 00:00:00 2001 From: shamoon <4887959+shamoon@users.noreply.github.com> Date: Thu, 5 Sep 2024 15:32:03 -0700 Subject: [PATCH] Enhancement: re-work mail rule dialog, support multiple include patterns (#7635) --- docs/usage.md | 2 +- src-ui/messages.xlf | 145 +++++++++--------- .../mail-rule-edit-dialog.component.html | 65 +++++--- src/paperless_mail/mail.py | 25 ++- src/paperless_mail/tests/test_mail.py | 6 + 5 files changed, 143 insertions(+), 100 deletions(-) diff --git a/docs/usage.md b/docs/usage.md index bf45d4982..4a34c95f1 100644 --- a/docs/usage.md +++ b/docs/usage.md @@ -137,7 +137,7 @@ These rules perform the following: Paperless will check all emails only once and completely ignore messages that do not match your filters. It will also only perform the rule action on e-mails that it has consumed documents from. The filename attachment -exclusion pattern can include multiple patterns separated by a comma. +patterns can include wildcards and multiple patterns separated by a comma. The actions all ensure that the same mail is not consumed twice by different means. These are as follows: diff --git a/src-ui/messages.xlf b/src-ui/messages.xlf index 161ff52d7..99831c7e4 100644 --- a/src-ui/messages.xlf +++ b/src-ui/messages.xlf @@ -560,7 +560,7 @@ src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 51 + 72 src/app/components/common/edit-dialog/storage-path-edit-dialog/storage-path-edit-dialog.component.html @@ -1648,7 +1648,7 @@ src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 50 + 71 src/app/components/common/edit-dialog/storage-path-edit-dialog/storage-path-edit-dialog.component.html @@ -3650,162 +3650,166 @@ 88 + + Rule order + + src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html + 16 + + Account src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 14 + 19 src/app/components/manage/mail/mail.component.html 68 + + Paperless will only process mails that match all of the criteria specified below. + + src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html + 24 + + Folder src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 15 + 26 Subfolders must be separated by a delimiter, often a dot ('.') or slash ('/'), but it varies by mail server. src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 15 + 26 Maximum age (days) src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 16 - - - - Attachment type - - src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 17 - - - - Consumption scope - - src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 18 - - - - See docs for .eml processing requirements - - src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 18 - - - - Rule order - - src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 19 - - - - Paperless will only process mails that match all of the filters specified below. - - src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 22 + 27 Filter from src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 23 + 30 Filter to src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 24 + 31 Filter subject src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 25 + 32 Filter body src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 26 + 33 - - Filter attachment filename includes + + Consumption scope src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 27 + 39 - - Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive. + + See docs for .eml processing requirements src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 27 + 39 - - Filter attachment filename excluding + + Attachment type src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 28 + 40 - - Do not consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive. + + Include only files matching src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 28 + 43 + + + + Optional. Wildcards e.g. *.pdf or *invoice* allowed. Can be comma-separated list. Case insensitive. + + src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html + 43 + + + src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html + 44 + + + + Exclude files matching + + src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html + 44 Action src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 31 + 50 - - Action is only performed when documents are consumed from the mail. Mails without attachments remain entirely untouched. + + Only performed if the mail is processed. src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 31 + 50 Action parameter src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 33 + 52 Assign title from src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 35 + 54 + + + + Assign owner from rule + + src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html + 55 Assign document type src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 37 + 59 src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html @@ -3816,32 +3820,25 @@ Assign correspondent from src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 38 + 60 Assign correspondent src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 40 + 62 src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html 163 - - Assign owner from rule - - src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 42 - - Error src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html - 48 + 69 src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html diff --git a/src-ui/src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html b/src-ui/src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html index d9c3ed717..a7c0617b0 100644 --- a/src-ui/src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html +++ b/src-ui/src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html @@ -10,36 +10,57 @@ diff --git a/src/paperless_mail/mail.py b/src/paperless_mail/mail.py index 92c471845..4ecd44659 100644 --- a/src/paperless_mail/mail.py +++ b/src/paperless_mail/mail.py @@ -686,6 +686,25 @@ class MailAccountHandler(LoggingMixin): return processed_elements + def filename_inclusion_matches( + self, + filter_attachment_filename_include: Optional[str], + filename: str, + ) -> bool: + if filter_attachment_filename_include: + filter_attachment_filename_inclusions = ( + filter_attachment_filename_include.split(",") + ) + + # Force the filename and pattern to the lowercase + # as this is system dependent otherwise + filename = filename.lower() + for filename_include in filter_attachment_filename_inclusions: + if filename_include and fnmatch(filename, filename_include.lower()): + return True + return False + return True + def filename_exclusion_matches( self, filter_attachment_filename_exclude: Optional[str], @@ -728,9 +747,9 @@ class MailAccountHandler(LoggingMixin): ) continue - if rule.filter_attachment_filename_include and not fnmatch( - att.filename.lower(), - rule.filter_attachment_filename_include.lower(), + if not self.filename_inclusion_matches( + rule.filter_attachment_filename_include, + att.filename, ): # Force the filename and pattern to the lowercase # as this is system dependent otherwise diff --git a/src/paperless_mail/tests/test_mail.py b/src/paperless_mail/tests/test_mail.py index d671021bf..ef359db75 100644 --- a/src/paperless_mail/tests/test_mail.py +++ b/src/paperless_mail/tests/test_mail.py @@ -658,6 +658,12 @@ class TestMail( exclude_pattern=None, expected_matches=["f2.png"], ), + FilterTestCase( + "PDF Files with f2 and f3", + include_pattern="f2.pdf,f3*", + exclude_pattern=None, + expected_matches=["f2.pdf", "f3.pdf"], + ), FilterTestCase( "PDF Files without f1", include_pattern="*.pdf",