mirror of
				https://github.com/paperless-ngx/paperless-ngx.git
				synced 2025-10-26 08:12:34 -04:00 
			
		
		
		
	Chore: Unify workflow logic (#7880)
This commit is contained in:
		
							parent
							
								
									024b60638a
								
							
						
					
					
						commit
						dcc8d4046a
					
				| @ -12,8 +12,8 @@ class DocumentsConfig(AppConfig): | |||||||
|         from documents.signals import document_updated |         from documents.signals import document_updated | ||||||
|         from documents.signals.handlers import add_inbox_tags |         from documents.signals.handlers import add_inbox_tags | ||||||
|         from documents.signals.handlers import add_to_index |         from documents.signals.handlers import add_to_index | ||||||
|         from documents.signals.handlers import run_workflow_added |         from documents.signals.handlers import run_workflows_added | ||||||
|         from documents.signals.handlers import run_workflow_updated |         from documents.signals.handlers import run_workflows_updated | ||||||
|         from documents.signals.handlers import set_correspondent |         from documents.signals.handlers import set_correspondent | ||||||
|         from documents.signals.handlers import set_document_type |         from documents.signals.handlers import set_document_type | ||||||
|         from documents.signals.handlers import set_log_entry |         from documents.signals.handlers import set_log_entry | ||||||
| @ -27,7 +27,7 @@ class DocumentsConfig(AppConfig): | |||||||
|         document_consumption_finished.connect(set_storage_path) |         document_consumption_finished.connect(set_storage_path) | ||||||
|         document_consumption_finished.connect(set_log_entry) |         document_consumption_finished.connect(set_log_entry) | ||||||
|         document_consumption_finished.connect(add_to_index) |         document_consumption_finished.connect(add_to_index) | ||||||
|         document_consumption_finished.connect(run_workflow_added) |         document_consumption_finished.connect(run_workflows_added) | ||||||
|         document_updated.connect(run_workflow_updated) |         document_updated.connect(run_workflows_updated) | ||||||
| 
 | 
 | ||||||
|         AppConfig.ready(self) |         AppConfig.ready(self) | ||||||
|  | |||||||
| @ -4,7 +4,6 @@ import os | |||||||
| import tempfile | import tempfile | ||||||
| from enum import Enum | from enum import Enum | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
| from typing import TYPE_CHECKING |  | ||||||
| 
 | 
 | ||||||
| import magic | import magic | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| @ -21,7 +20,6 @@ from documents.data_models import DocumentMetadataOverrides | |||||||
| from documents.file_handling import create_source_path_directory | from documents.file_handling import create_source_path_directory | ||||||
| from documents.file_handling import generate_unique_filename | from documents.file_handling import generate_unique_filename | ||||||
| from documents.loggers import LoggingMixin | from documents.loggers import LoggingMixin | ||||||
| from documents.matching import document_matches_workflow |  | ||||||
| from documents.models import Correspondent | from documents.models import Correspondent | ||||||
| from documents.models import CustomField | from documents.models import CustomField | ||||||
| from documents.models import CustomFieldInstance | from documents.models import CustomFieldInstance | ||||||
| @ -30,8 +28,6 @@ from documents.models import DocumentType | |||||||
| from documents.models import FileInfo | from documents.models import FileInfo | ||||||
| from documents.models import StoragePath | from documents.models import StoragePath | ||||||
| from documents.models import Tag | from documents.models import Tag | ||||||
| from documents.models import Workflow |  | ||||||
| from documents.models import WorkflowAction |  | ||||||
| from documents.models import WorkflowTrigger | from documents.models import WorkflowTrigger | ||||||
| from documents.parsers import DocumentParser | from documents.parsers import DocumentParser | ||||||
| from documents.parsers import ParseError | from documents.parsers import ParseError | ||||||
| @ -46,6 +42,8 @@ from documents.plugins.helpers import ProgressManager | |||||||
| from documents.plugins.helpers import ProgressStatusOptions | from documents.plugins.helpers import ProgressStatusOptions | ||||||
| from documents.signals import document_consumption_finished | from documents.signals import document_consumption_finished | ||||||
| from documents.signals import document_consumption_started | from documents.signals import document_consumption_started | ||||||
|  | from documents.signals.handlers import run_workflows | ||||||
|  | from documents.templating.title import parse_doc_title_w_placeholders | ||||||
| from documents.utils import copy_basic_file_stats | from documents.utils import copy_basic_file_stats | ||||||
| from documents.utils import copy_file_with_basic_stats | from documents.utils import copy_file_with_basic_stats | ||||||
| from documents.utils import run_subprocess | from documents.utils import run_subprocess | ||||||
| @ -63,162 +61,13 @@ class WorkflowTriggerPlugin( | |||||||
|         """ |         """ | ||||||
|         Get overrides from matching workflows |         Get overrides from matching workflows | ||||||
|         """ |         """ | ||||||
|         msg = "" |         overrides, msg = run_workflows( | ||||||
|         overrides = DocumentMetadataOverrides() |  | ||||||
|         for workflow in ( |  | ||||||
|             Workflow.objects.filter(enabled=True) |  | ||||||
|             .prefetch_related("actions") |  | ||||||
|             .prefetch_related("actions__assign_view_users") |  | ||||||
|             .prefetch_related("actions__assign_view_groups") |  | ||||||
|             .prefetch_related("actions__assign_change_users") |  | ||||||
|             .prefetch_related("actions__assign_change_groups") |  | ||||||
|             .prefetch_related("actions__assign_custom_fields") |  | ||||||
|             .prefetch_related("actions__remove_tags") |  | ||||||
|             .prefetch_related("actions__remove_correspondents") |  | ||||||
|             .prefetch_related("actions__remove_document_types") |  | ||||||
|             .prefetch_related("actions__remove_storage_paths") |  | ||||||
|             .prefetch_related("actions__remove_custom_fields") |  | ||||||
|             .prefetch_related("actions__remove_owners") |  | ||||||
|             .prefetch_related("triggers") |  | ||||||
|             .order_by("order") |  | ||||||
|         ): |  | ||||||
|             action_overrides = DocumentMetadataOverrides() |  | ||||||
| 
 |  | ||||||
|             if document_matches_workflow( |  | ||||||
|                 self.input_doc, |  | ||||||
|                 workflow, |  | ||||||
|             WorkflowTrigger.WorkflowTriggerType.CONSUMPTION, |             WorkflowTrigger.WorkflowTriggerType.CONSUMPTION, | ||||||
|             ): |             self.input_doc, | ||||||
|                 for action in workflow.actions.all(): |             None, | ||||||
|                     if TYPE_CHECKING: |             DocumentMetadataOverrides(), | ||||||
|                         assert isinstance(action, WorkflowAction) |  | ||||||
|                     msg += f"Applying {action} from {workflow}\n" |  | ||||||
|                     if action.type == WorkflowAction.WorkflowActionType.ASSIGNMENT: |  | ||||||
|                         if action.assign_title is not None: |  | ||||||
|                             action_overrides.title = action.assign_title |  | ||||||
|                         if action.assign_tags is not None: |  | ||||||
|                             action_overrides.tag_ids = list( |  | ||||||
|                                 action.assign_tags.values_list("pk", flat=True), |  | ||||||
|         ) |         ) | ||||||
| 
 |         if overrides: | ||||||
|                         if action.assign_correspondent is not None: |  | ||||||
|                             action_overrides.correspondent_id = ( |  | ||||||
|                                 action.assign_correspondent.pk |  | ||||||
|                             ) |  | ||||||
|                         if action.assign_document_type is not None: |  | ||||||
|                             action_overrides.document_type_id = ( |  | ||||||
|                                 action.assign_document_type.pk |  | ||||||
|                             ) |  | ||||||
|                         if action.assign_storage_path is not None: |  | ||||||
|                             action_overrides.storage_path_id = ( |  | ||||||
|                                 action.assign_storage_path.pk |  | ||||||
|                             ) |  | ||||||
|                         if action.assign_owner is not None: |  | ||||||
|                             action_overrides.owner_id = action.assign_owner.pk |  | ||||||
|                         if action.assign_view_users is not None: |  | ||||||
|                             action_overrides.view_users = list( |  | ||||||
|                                 action.assign_view_users.values_list("pk", flat=True), |  | ||||||
|                             ) |  | ||||||
|                         if action.assign_view_groups is not None: |  | ||||||
|                             action_overrides.view_groups = list( |  | ||||||
|                                 action.assign_view_groups.values_list("pk", flat=True), |  | ||||||
|                             ) |  | ||||||
|                         if action.assign_change_users is not None: |  | ||||||
|                             action_overrides.change_users = list( |  | ||||||
|                                 action.assign_change_users.values_list("pk", flat=True), |  | ||||||
|                             ) |  | ||||||
|                         if action.assign_change_groups is not None: |  | ||||||
|                             action_overrides.change_groups = list( |  | ||||||
|                                 action.assign_change_groups.values_list( |  | ||||||
|                                     "pk", |  | ||||||
|                                     flat=True, |  | ||||||
|                                 ), |  | ||||||
|                             ) |  | ||||||
|                         if action.assign_custom_fields is not None: |  | ||||||
|                             action_overrides.custom_field_ids = list( |  | ||||||
|                                 action.assign_custom_fields.values_list( |  | ||||||
|                                     "pk", |  | ||||||
|                                     flat=True, |  | ||||||
|                                 ), |  | ||||||
|                             ) |  | ||||||
|                         overrides.update(action_overrides) |  | ||||||
|                     elif action.type == WorkflowAction.WorkflowActionType.REMOVAL: |  | ||||||
|                         # Removal actions overwrite the current overrides |  | ||||||
|                         if action.remove_all_tags: |  | ||||||
|                             overrides.tag_ids = [] |  | ||||||
|                         elif overrides.tag_ids: |  | ||||||
|                             for tag in action.remove_custom_fields.filter( |  | ||||||
|                                 pk__in=overrides.tag_ids, |  | ||||||
|                             ): |  | ||||||
|                                 overrides.tag_ids.remove(tag.pk) |  | ||||||
| 
 |  | ||||||
|                         if action.remove_all_correspondents or ( |  | ||||||
|                             overrides.correspondent_id is not None |  | ||||||
|                             and action.remove_correspondents.filter( |  | ||||||
|                                 pk=overrides.correspondent_id, |  | ||||||
|                             ).exists() |  | ||||||
|                         ): |  | ||||||
|                             overrides.correspondent_id = None |  | ||||||
| 
 |  | ||||||
|                         if action.remove_all_document_types or ( |  | ||||||
|                             overrides.document_type_id is not None |  | ||||||
|                             and action.remove_document_types.filter( |  | ||||||
|                                 pk=overrides.document_type_id, |  | ||||||
|                             ).exists() |  | ||||||
|                         ): |  | ||||||
|                             overrides.document_type_id = None |  | ||||||
| 
 |  | ||||||
|                         if action.remove_all_storage_paths or ( |  | ||||||
|                             overrides.storage_path_id is not None |  | ||||||
|                             and action.remove_storage_paths.filter( |  | ||||||
|                                 pk=overrides.storage_path_id, |  | ||||||
|                             ).exists() |  | ||||||
|                         ): |  | ||||||
|                             overrides.storage_path_id = None |  | ||||||
| 
 |  | ||||||
|                         if action.remove_all_custom_fields: |  | ||||||
|                             overrides.custom_field_ids = [] |  | ||||||
|                         elif overrides.custom_field_ids: |  | ||||||
|                             for field in action.remove_custom_fields.filter( |  | ||||||
|                                 pk__in=overrides.custom_field_ids, |  | ||||||
|                             ): |  | ||||||
|                                 overrides.custom_field_ids.remove(field.pk) |  | ||||||
| 
 |  | ||||||
|                         if action.remove_all_owners or ( |  | ||||||
|                             overrides.owner_id is not None |  | ||||||
|                             and action.remove_owners.filter( |  | ||||||
|                                 pk=overrides.owner_id, |  | ||||||
|                             ).exists() |  | ||||||
|                         ): |  | ||||||
|                             overrides.owner_id = None |  | ||||||
| 
 |  | ||||||
|                         if action.remove_all_permissions: |  | ||||||
|                             overrides.view_users = [] |  | ||||||
|                             overrides.view_groups = [] |  | ||||||
|                             overrides.change_users = [] |  | ||||||
|                             overrides.change_groups = [] |  | ||||||
|                         else: |  | ||||||
|                             if overrides.view_users: |  | ||||||
|                                 for user in action.remove_view_users.filter( |  | ||||||
|                                     pk__in=overrides.view_users, |  | ||||||
|                                 ): |  | ||||||
|                                     overrides.view_users.remove(user.pk) |  | ||||||
|                             if overrides.change_users: |  | ||||||
|                                 for user in action.remove_change_users.filter( |  | ||||||
|                                     pk__in=overrides.change_users, |  | ||||||
|                                 ): |  | ||||||
|                                     overrides.change_users.remove(user.pk) |  | ||||||
|                             if overrides.view_groups: |  | ||||||
|                                 for user in action.remove_view_groups.filter( |  | ||||||
|                                     pk__in=overrides.view_groups, |  | ||||||
|                                 ): |  | ||||||
|                                     overrides.view_groups.remove(user.pk) |  | ||||||
|                             if overrides.change_groups: |  | ||||||
|                                 for user in action.remove_change_groups.filter( |  | ||||||
|                                     pk__in=overrides.change_groups, |  | ||||||
|                                 ): |  | ||||||
|                                     overrides.change_groups.remove(user.pk) |  | ||||||
| 
 |  | ||||||
|             self.metadata.update(overrides) |             self.metadata.update(overrides) | ||||||
|         return msg |         return msg | ||||||
| 
 | 
 | ||||||
| @ -948,47 +797,3 @@ class ConsumerPlugin( | |||||||
|             copy_basic_file_stats(source, target) |             copy_basic_file_stats(source, target) | ||||||
|         except Exception:  # pragma: no cover |         except Exception:  # pragma: no cover | ||||||
|             pass |             pass | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| def parse_doc_title_w_placeholders( |  | ||||||
|     title: str, |  | ||||||
|     correspondent_name: str, |  | ||||||
|     doc_type_name: str, |  | ||||||
|     owner_username: str, |  | ||||||
|     local_added: datetime.datetime, |  | ||||||
|     original_filename: str, |  | ||||||
|     created: datetime.datetime | None = None, |  | ||||||
| ) -> str: |  | ||||||
|     """ |  | ||||||
|     Available title placeholders for Workflows depend on what has already been assigned, |  | ||||||
|     e.g. for pre-consumption triggers created will not have been parsed yet, but it will |  | ||||||
|     for added / updated triggers |  | ||||||
|     """ |  | ||||||
|     formatting = { |  | ||||||
|         "correspondent": correspondent_name, |  | ||||||
|         "document_type": doc_type_name, |  | ||||||
|         "added": local_added.isoformat(), |  | ||||||
|         "added_year": local_added.strftime("%Y"), |  | ||||||
|         "added_year_short": local_added.strftime("%y"), |  | ||||||
|         "added_month": local_added.strftime("%m"), |  | ||||||
|         "added_month_name": local_added.strftime("%B"), |  | ||||||
|         "added_month_name_short": local_added.strftime("%b"), |  | ||||||
|         "added_day": local_added.strftime("%d"), |  | ||||||
|         "added_time": local_added.strftime("%H:%M"), |  | ||||||
|         "owner_username": owner_username, |  | ||||||
|         "original_filename": Path(original_filename).stem, |  | ||||||
|     } |  | ||||||
|     if created is not None: |  | ||||||
|         formatting.update( |  | ||||||
|             { |  | ||||||
|                 "created": created.isoformat(), |  | ||||||
|                 "created_year": created.strftime("%Y"), |  | ||||||
|                 "created_year_short": created.strftime("%y"), |  | ||||||
|                 "created_month": created.strftime("%m"), |  | ||||||
|                 "created_month_name": created.strftime("%B"), |  | ||||||
|                 "created_month_name_short": created.strftime("%b"), |  | ||||||
|                 "created_day": created.strftime("%d"), |  | ||||||
|                 "created_time": created.strftime("%H:%M"), |  | ||||||
|             }, |  | ||||||
|         ) |  | ||||||
|     return title.format(**formatting).strip() |  | ||||||
|  | |||||||
| @ -24,7 +24,8 @@ from guardian.shortcuts import remove_perm | |||||||
| from documents import matching | from documents import matching | ||||||
| from documents.caching import clear_document_caches | from documents.caching import clear_document_caches | ||||||
| from documents.classifier import DocumentClassifier | from documents.classifier import DocumentClassifier | ||||||
| from documents.consumer import parse_doc_title_w_placeholders | from documents.data_models import ConsumableDocument | ||||||
|  | from documents.data_models import DocumentMetadataOverrides | ||||||
| from documents.file_handling import create_source_path_directory | from documents.file_handling import create_source_path_directory | ||||||
| from documents.file_handling import delete_empty_directories | from documents.file_handling import delete_empty_directories | ||||||
| from documents.file_handling import generate_unique_filename | from documents.file_handling import generate_unique_filename | ||||||
| @ -38,6 +39,7 @@ from documents.models import WorkflowAction | |||||||
| from documents.models import WorkflowTrigger | from documents.models import WorkflowTrigger | ||||||
| from documents.permissions import get_objects_for_user_owner_aware | from documents.permissions import get_objects_for_user_owner_aware | ||||||
| from documents.permissions import set_permissions_for_object | from documents.permissions import set_permissions_for_object | ||||||
|  | from documents.templating.title import parse_doc_title_w_placeholders | ||||||
| 
 | 
 | ||||||
| logger = logging.getLogger("paperless.handlers") | logger = logging.getLogger("paperless.handlers") | ||||||
| 
 | 
 | ||||||
| @ -511,66 +513,81 @@ def add_to_index(sender, document, **kwargs): | |||||||
|     index.add_or_update_document(document) |     index.add_or_update_document(document) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def run_workflow_added(sender, document: Document, logging_group=None, **kwargs): | def run_workflows_added(sender, document: Document, logging_group=None, **kwargs): | ||||||
|     run_workflow( |     run_workflows( | ||||||
|         WorkflowTrigger.WorkflowTriggerType.DOCUMENT_ADDED, |         WorkflowTrigger.WorkflowTriggerType.DOCUMENT_ADDED, | ||||||
|         document, |         document, | ||||||
|         logging_group, |         logging_group, | ||||||
|     ) |     ) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def run_workflow_updated(sender, document: Document, logging_group=None, **kwargs): | def run_workflows_updated(sender, document: Document, logging_group=None, **kwargs): | ||||||
|     run_workflow( |     run_workflows( | ||||||
|         WorkflowTrigger.WorkflowTriggerType.DOCUMENT_UPDATED, |         WorkflowTrigger.WorkflowTriggerType.DOCUMENT_UPDATED, | ||||||
|         document, |         document, | ||||||
|         logging_group, |         logging_group, | ||||||
|     ) |     ) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def run_workflow( | def run_workflows( | ||||||
|     trigger_type: WorkflowTrigger.WorkflowTriggerType, |     trigger_type: WorkflowTrigger.WorkflowTriggerType, | ||||||
|     document: Document, |     document: Document | ConsumableDocument, | ||||||
|     logging_group=None, |     logging_group=None, | ||||||
| ): |     overrides: DocumentMetadataOverrides | None = None, | ||||||
|  | ) -> tuple[DocumentMetadataOverrides, str] | None: | ||||||
|  |     """Run workflows which match a Document (or ConsumableDocument) for a specific trigger type. | ||||||
|  | 
 | ||||||
|  |     Assignment or removal actions are either applied directly to the document or an overrides object. If an overrides | ||||||
|  |     object is provided, the function returns the object with the applied changes or None if no actions were applied and a string | ||||||
|  |     of messages for each action. If no overrides object is provided, the changes are applied directly to the document and the | ||||||
|  |     function returns None. | ||||||
|  |     """ | ||||||
|  | 
 | ||||||
|     def assignment_action(): |     def assignment_action(): | ||||||
|         if action.assign_tags.all().count() > 0: |         if action.assign_tags.exists(): | ||||||
|             doc_tag_ids.extend( |             if not use_overrides: | ||||||
|                 list(action.assign_tags.all().values_list("pk", flat=True)), |                 doc_tag_ids.extend(action.assign_tags.values_list("pk", flat=True)) | ||||||
|  |             else: | ||||||
|  |                 if overrides.tag_ids is None: | ||||||
|  |                     overrides.tag_ids = [] | ||||||
|  |                 overrides.tag_ids.extend( | ||||||
|  |                     action.assign_tags.values_list("pk", flat=True), | ||||||
|                 ) |                 ) | ||||||
| 
 | 
 | ||||||
|         if action.assign_correspondent is not None: |         if action.assign_correspondent: | ||||||
|  |             if not use_overrides: | ||||||
|                 document.correspondent = action.assign_correspondent |                 document.correspondent = action.assign_correspondent | ||||||
|  |             else: | ||||||
|  |                 overrides.correspondent_id = action.assign_correspondent.pk | ||||||
| 
 | 
 | ||||||
|         if action.assign_document_type is not None: |         if action.assign_document_type: | ||||||
|  |             if not use_overrides: | ||||||
|                 document.document_type = action.assign_document_type |                 document.document_type = action.assign_document_type | ||||||
|  |             else: | ||||||
|  |                 overrides.document_type_id = action.assign_document_type.pk | ||||||
| 
 | 
 | ||||||
|         if action.assign_storage_path is not None: |         if action.assign_storage_path: | ||||||
|  |             if not use_overrides: | ||||||
|                 document.storage_path = action.assign_storage_path |                 document.storage_path = action.assign_storage_path | ||||||
|  |             else: | ||||||
|  |                 overrides.storage_path_id = action.assign_storage_path.pk | ||||||
| 
 | 
 | ||||||
|         if action.assign_owner is not None: |         if action.assign_owner: | ||||||
|  |             if not use_overrides: | ||||||
|                 document.owner = action.assign_owner |                 document.owner = action.assign_owner | ||||||
|  |             else: | ||||||
|  |                 overrides.owner_id = action.assign_owner.pk | ||||||
| 
 | 
 | ||||||
|         if action.assign_title is not None: |         if action.assign_title: | ||||||
|  |             if not use_overrides: | ||||||
|                 try: |                 try: | ||||||
|                     document.title = parse_doc_title_w_placeholders( |                     document.title = parse_doc_title_w_placeholders( | ||||||
|                         action.assign_title, |                         action.assign_title, | ||||||
|                     ( |                         document.correspondent.name if document.correspondent else "", | ||||||
|                         document.correspondent.name |                         document.document_type.name if document.document_type else "", | ||||||
|                         if document.correspondent is not None |                         document.owner.username if document.owner else "", | ||||||
|                         else "" |  | ||||||
|                     ), |  | ||||||
|                     ( |  | ||||||
|                         document.document_type.name |  | ||||||
|                         if document.document_type is not None |  | ||||||
|                         else "" |  | ||||||
|                     ), |  | ||||||
|                     (document.owner.username if document.owner is not None else ""), |  | ||||||
|                         timezone.localtime(document.added), |                         timezone.localtime(document.added), | ||||||
|                     ( |                         document.original_filename or "", | ||||||
|                         document.original_filename |  | ||||||
|                         if document.original_filename is not None |  | ||||||
|                         else "" |  | ||||||
|                     ), |  | ||||||
|                         timezone.localtime(document.created), |                         timezone.localtime(document.created), | ||||||
|                     ) |                     ) | ||||||
|                 except Exception: |                 except Exception: | ||||||
| @ -578,135 +595,202 @@ def run_workflow( | |||||||
|                         f"Error occurred parsing title assignment '{action.assign_title}', falling back to original", |                         f"Error occurred parsing title assignment '{action.assign_title}', falling back to original", | ||||||
|                         extra={"group": logging_group}, |                         extra={"group": logging_group}, | ||||||
|                     ) |                     ) | ||||||
|  |             else: | ||||||
|  |                 overrides.title = action.assign_title | ||||||
| 
 | 
 | ||||||
|         if ( |         if any( | ||||||
|             ( |             [ | ||||||
|                 action.assign_view_users is not None |                 action.assign_view_users.exists(), | ||||||
|                 and action.assign_view_users.count() > 0 |                 action.assign_view_groups.exists(), | ||||||
|             ) |                 action.assign_change_users.exists(), | ||||||
|             or ( |                 action.assign_change_groups.exists(), | ||||||
|                 action.assign_view_groups is not None |             ], | ||||||
|                 and action.assign_view_groups.count() > 0 |  | ||||||
|             ) |  | ||||||
|             or ( |  | ||||||
|                 action.assign_change_users is not None |  | ||||||
|                 and action.assign_change_users.count() > 0 |  | ||||||
|             ) |  | ||||||
|             or ( |  | ||||||
|                 action.assign_change_groups is not None |  | ||||||
|                 and action.assign_change_groups.count() > 0 |  | ||||||
|             ) |  | ||||||
|         ): |         ): | ||||||
|             permissions = { |             permissions = { | ||||||
|                 "view": { |                 "view": { | ||||||
|                     "users": action.assign_view_users.all().values_list( |                     "users": action.assign_view_users.values_list("id", flat=True), | ||||||
|                         "id", |                     "groups": action.assign_view_groups.values_list("id", flat=True), | ||||||
|                     ) |  | ||||||
|                     or [], |  | ||||||
|                     "groups": action.assign_view_groups.all().values_list( |  | ||||||
|                         "id", |  | ||||||
|                     ) |  | ||||||
|                     or [], |  | ||||||
|                 }, |                 }, | ||||||
|                 "change": { |                 "change": { | ||||||
|                     "users": action.assign_change_users.all().values_list( |                     "users": action.assign_change_users.values_list("id", flat=True), | ||||||
|                         "id", |                     "groups": action.assign_change_groups.values_list("id", flat=True), | ||||||
|                     ) |  | ||||||
|                     or [], |  | ||||||
|                     "groups": action.assign_change_groups.all().values_list( |  | ||||||
|                         "id", |  | ||||||
|                     ) |  | ||||||
|                     or [], |  | ||||||
|                 }, |                 }, | ||||||
|             } |             } | ||||||
|  |             if not use_overrides: | ||||||
|                 set_permissions_for_object( |                 set_permissions_for_object( | ||||||
|                     permissions=permissions, |                     permissions=permissions, | ||||||
|                     object=document, |                     object=document, | ||||||
|                     merge=True, |                     merge=True, | ||||||
|                 ) |                 ) | ||||||
|  |             else: | ||||||
|  |                 overrides.view_users = list( | ||||||
|  |                     set( | ||||||
|  |                         (overrides.view_users or []) | ||||||
|  |                         + list(permissions["view"]["users"]), | ||||||
|  |                     ), | ||||||
|  |                 ) | ||||||
|  |                 overrides.view_groups = list( | ||||||
|  |                     set( | ||||||
|  |                         (overrides.view_groups or []) | ||||||
|  |                         + list(permissions["view"]["groups"]), | ||||||
|  |                     ), | ||||||
|  |                 ) | ||||||
|  |                 overrides.change_users = list( | ||||||
|  |                     set( | ||||||
|  |                         (overrides.change_users or []) | ||||||
|  |                         + list(permissions["change"]["users"]), | ||||||
|  |                     ), | ||||||
|  |                 ) | ||||||
|  |                 overrides.change_groups = list( | ||||||
|  |                     set( | ||||||
|  |                         (overrides.change_groups or []) | ||||||
|  |                         + list(permissions["change"]["groups"]), | ||||||
|  |                     ), | ||||||
|  |                 ) | ||||||
| 
 | 
 | ||||||
|         if action.assign_custom_fields is not None: |         if action.assign_custom_fields.exists(): | ||||||
|  |             if not use_overrides: | ||||||
|                 for field in action.assign_custom_fields.all(): |                 for field in action.assign_custom_fields.all(): | ||||||
|                 if ( |                     if not CustomFieldInstance.objects.filter( | ||||||
|                     CustomFieldInstance.objects.filter( |  | ||||||
|                         field=field, |                         field=field, | ||||||
|                         document=document, |                         document=document, | ||||||
|                     ).count() |                     ).exists(): | ||||||
|                     == 0 |  | ||||||
|                 ): |  | ||||||
|                         # can be triggered on existing docs, so only add the field if it doesn't already exist |                         # can be triggered on existing docs, so only add the field if it doesn't already exist | ||||||
|                         CustomFieldInstance.objects.create( |                         CustomFieldInstance.objects.create( | ||||||
|                             field=field, |                             field=field, | ||||||
|                             document=document, |                             document=document, | ||||||
|                         ) |                         ) | ||||||
|  |             else: | ||||||
|  |                 overrides.custom_field_ids = list( | ||||||
|  |                     set( | ||||||
|  |                         (overrides.custom_field_ids or []) | ||||||
|  |                         + list( | ||||||
|  |                             action.assign_custom_fields.values_list("pk", flat=True), | ||||||
|  |                         ), | ||||||
|  |                     ), | ||||||
|  |                 ) | ||||||
| 
 | 
 | ||||||
|     def removal_action(): |     def removal_action(): | ||||||
|         if action.remove_all_tags: |         if action.remove_all_tags: | ||||||
|  |             if not use_overrides: | ||||||
|                 doc_tag_ids.clear() |                 doc_tag_ids.clear() | ||||||
|             else: |             else: | ||||||
|  |                 overrides.tag_ids = None | ||||||
|  |         else: | ||||||
|  |             if not use_overrides: | ||||||
|                 for tag in action.remove_tags.filter( |                 for tag in action.remove_tags.filter( | ||||||
|                 pk__in=list(document.tags.values_list("pk", flat=True)), |                     pk__in=document.tags.values_list("pk", flat=True), | ||||||
|             ).all(): |                 ): | ||||||
|                     doc_tag_ids.remove(tag.pk) |                     doc_tag_ids.remove(tag.pk) | ||||||
|  |             elif overrides.tag_ids: | ||||||
|  |                 for tag in action.remove_tags.filter(pk__in=overrides.tag_ids): | ||||||
|  |                     overrides.tag_ids.remove(tag.pk) | ||||||
| 
 | 
 | ||||||
|         if action.remove_all_correspondents or ( |         if not use_overrides and ( | ||||||
|  |             action.remove_all_correspondents | ||||||
|  |             or ( | ||||||
|                 document.correspondent |                 document.correspondent | ||||||
|             and ( |                 and action.remove_correspondents.filter( | ||||||
|                 action.remove_correspondents.filter( |  | ||||||
|                     pk=document.correspondent.pk, |                     pk=document.correspondent.pk, | ||||||
|                 ).exists() |                 ).exists() | ||||||
|             ) |             ) | ||||||
|         ): |         ): | ||||||
|             document.correspondent = None |             document.correspondent = None | ||||||
|  |         elif use_overrides and ( | ||||||
|  |             action.remove_all_correspondents | ||||||
|  |             or ( | ||||||
|  |                 overrides.correspondent_id | ||||||
|  |                 and action.remove_correspondents.filter( | ||||||
|  |                     pk=overrides.correspondent_id, | ||||||
|  |                 ).exists() | ||||||
|  |             ) | ||||||
|  |         ): | ||||||
|  |             overrides.correspondent_id = None | ||||||
| 
 | 
 | ||||||
|         if action.remove_all_document_types or ( |         if not use_overrides and ( | ||||||
|  |             action.remove_all_document_types | ||||||
|  |             or ( | ||||||
|                 document.document_type |                 document.document_type | ||||||
|             and ( |                 and action.remove_document_types.filter( | ||||||
|                 action.remove_document_types.filter( |  | ||||||
|                     pk=document.document_type.pk, |                     pk=document.document_type.pk, | ||||||
|                 ).exists() |                 ).exists() | ||||||
|             ) |             ) | ||||||
|         ): |         ): | ||||||
|             document.document_type = None |             document.document_type = None | ||||||
|  |         elif use_overrides and ( | ||||||
|  |             action.remove_all_document_types | ||||||
|  |             or ( | ||||||
|  |                 overrides.document_type_id | ||||||
|  |                 and action.remove_document_types.filter( | ||||||
|  |                     pk=overrides.document_type_id, | ||||||
|  |                 ).exists() | ||||||
|  |             ) | ||||||
|  |         ): | ||||||
|  |             overrides.document_type_id = None | ||||||
| 
 | 
 | ||||||
|         if action.remove_all_storage_paths or ( |         if not use_overrides and ( | ||||||
|  |             action.remove_all_storage_paths | ||||||
|  |             or ( | ||||||
|                 document.storage_path |                 document.storage_path | ||||||
|             and ( |                 and action.remove_storage_paths.filter( | ||||||
|                 action.remove_storage_paths.filter( |  | ||||||
|                     pk=document.storage_path.pk, |                     pk=document.storage_path.pk, | ||||||
|                 ).exists() |                 ).exists() | ||||||
|             ) |             ) | ||||||
|         ): |         ): | ||||||
|             document.storage_path = None |             document.storage_path = None | ||||||
|  |         elif use_overrides and ( | ||||||
|  |             action.remove_all_storage_paths | ||||||
|  |             or ( | ||||||
|  |                 overrides.storage_path_id | ||||||
|  |                 and action.remove_storage_paths.filter( | ||||||
|  |                     pk=overrides.storage_path_id, | ||||||
|  |                 ).exists() | ||||||
|  |             ) | ||||||
|  |         ): | ||||||
|  |             overrides.storage_path_id = None | ||||||
| 
 | 
 | ||||||
|         if action.remove_all_owners or ( |         if not use_overrides and ( | ||||||
|  |             action.remove_all_owners | ||||||
|  |             or ( | ||||||
|                 document.owner |                 document.owner | ||||||
|             and (action.remove_owners.filter(pk=document.owner.pk).exists()) |                 and action.remove_owners.filter(pk=document.owner.pk).exists() | ||||||
|  |             ) | ||||||
|         ): |         ): | ||||||
|             document.owner = None |             document.owner = None | ||||||
|  |         elif use_overrides and ( | ||||||
|  |             action.remove_all_owners | ||||||
|  |             or ( | ||||||
|  |                 overrides.owner_id | ||||||
|  |                 and action.remove_owners.filter(pk=overrides.owner_id).exists() | ||||||
|  |             ) | ||||||
|  |         ): | ||||||
|  |             overrides.owner_id = None | ||||||
| 
 | 
 | ||||||
|         if action.remove_all_permissions: |         if action.remove_all_permissions: | ||||||
|  |             if not use_overrides: | ||||||
|                 permissions = { |                 permissions = { | ||||||
|                 "view": { |                     "view": {"users": [], "groups": []}, | ||||||
|                     "users": [], |                     "change": {"users": [], "groups": []}, | ||||||
|                     "groups": [], |  | ||||||
|                 }, |  | ||||||
|                 "change": { |  | ||||||
|                     "users": [], |  | ||||||
|                     "groups": [], |  | ||||||
|                 }, |  | ||||||
|                 } |                 } | ||||||
|                 set_permissions_for_object( |                 set_permissions_for_object( | ||||||
|                     permissions=permissions, |                     permissions=permissions, | ||||||
|                     object=document, |                     object=document, | ||||||
|                     merge=False, |                     merge=False, | ||||||
|                 ) |                 ) | ||||||
|         elif ( |             else: | ||||||
|             (action.remove_view_users.all().count() > 0) |                 overrides.view_users = None | ||||||
|             or (action.remove_view_groups.all().count() > 0) |                 overrides.view_groups = None | ||||||
|             or (action.remove_change_users.all().count() > 0) |                 overrides.change_users = None | ||||||
|             or (action.remove_change_groups.all().count() > 0) |                 overrides.change_groups = None | ||||||
|  |         elif any( | ||||||
|  |             [ | ||||||
|  |                 action.remove_view_users.exists(), | ||||||
|  |                 action.remove_view_groups.exists(), | ||||||
|  |                 action.remove_change_users.exists(), | ||||||
|  |                 action.remove_change_groups.exists(), | ||||||
|  |             ], | ||||||
|         ): |         ): | ||||||
|  |             if not use_overrides: | ||||||
|                 for user in action.remove_view_users.all(): |                 for user in action.remove_view_users.all(): | ||||||
|                     remove_perm("view_document", user, document) |                     remove_perm("view_document", user, document) | ||||||
|                 for user in action.remove_change_users.all(): |                 for user in action.remove_change_users.all(): | ||||||
| @ -715,62 +799,98 @@ def run_workflow( | |||||||
|                     remove_perm("view_document", group, document) |                     remove_perm("view_document", group, document) | ||||||
|                 for group in action.remove_change_groups.all(): |                 for group in action.remove_change_groups.all(): | ||||||
|                     remove_perm("change_document", group, document) |                     remove_perm("change_document", group, document) | ||||||
|  |             else: | ||||||
|  |                 if overrides.view_users: | ||||||
|  |                     for user in action.remove_view_users.filter( | ||||||
|  |                         pk__in=overrides.view_users, | ||||||
|  |                     ): | ||||||
|  |                         overrides.view_users.remove(user.pk) | ||||||
|  |                 if overrides.change_users: | ||||||
|  |                     for user in action.remove_change_users.filter( | ||||||
|  |                         pk__in=overrides.change_users, | ||||||
|  |                     ): | ||||||
|  |                         overrides.change_users.remove(user.pk) | ||||||
|  |                 if overrides.view_groups: | ||||||
|  |                     for group in action.remove_view_groups.filter( | ||||||
|  |                         pk__in=overrides.view_groups, | ||||||
|  |                     ): | ||||||
|  |                         overrides.view_groups.remove(group.pk) | ||||||
|  |                 if overrides.change_groups: | ||||||
|  |                     for group in action.remove_change_groups.filter( | ||||||
|  |                         pk__in=overrides.change_groups, | ||||||
|  |                     ): | ||||||
|  |                         overrides.change_groups.remove(group.pk) | ||||||
| 
 | 
 | ||||||
|         if action.remove_all_custom_fields: |         if action.remove_all_custom_fields: | ||||||
|  |             if not use_overrides: | ||||||
|                 CustomFieldInstance.objects.filter(document=document).delete() |                 CustomFieldInstance.objects.filter(document=document).delete() | ||||||
|         elif action.remove_custom_fields.all().count() > 0: |             else: | ||||||
|  |                 overrides.custom_field_ids = None | ||||||
|  |         elif action.remove_custom_fields.exists(): | ||||||
|  |             if not use_overrides: | ||||||
|                 CustomFieldInstance.objects.filter( |                 CustomFieldInstance.objects.filter( | ||||||
|                     field__in=action.remove_custom_fields.all(), |                     field__in=action.remove_custom_fields.all(), | ||||||
|                     document=document, |                     document=document, | ||||||
|                 ).delete() |                 ).delete() | ||||||
| 
 |             elif overrides.custom_field_ids: | ||||||
|     for workflow in ( |                 for field in action.remove_custom_fields.filter( | ||||||
|         Workflow.objects.filter( |                     pk__in=overrides.custom_field_ids, | ||||||
|             enabled=True, |  | ||||||
|             triggers__type=trigger_type, |  | ||||||
|         ) |  | ||||||
|         .prefetch_related("actions") |  | ||||||
|         .prefetch_related("actions__assign_view_users") |  | ||||||
|         .prefetch_related("actions__assign_view_groups") |  | ||||||
|         .prefetch_related("actions__assign_change_users") |  | ||||||
|         .prefetch_related("actions__assign_change_groups") |  | ||||||
|         .prefetch_related("actions__assign_custom_fields") |  | ||||||
|         .prefetch_related("actions__remove_tags") |  | ||||||
|         .prefetch_related("actions__remove_correspondents") |  | ||||||
|         .prefetch_related("actions__remove_document_types") |  | ||||||
|         .prefetch_related("actions__remove_storage_paths") |  | ||||||
|         .prefetch_related("actions__remove_custom_fields") |  | ||||||
|         .prefetch_related("actions__remove_owners") |  | ||||||
|         .prefetch_related("triggers") |  | ||||||
|         .order_by("order") |  | ||||||
|                 ): |                 ): | ||||||
|  |                     overrides.custom_field_ids.remove(field.pk) | ||||||
|  | 
 | ||||||
|  |     use_overrides = overrides is not None | ||||||
|  |     messages = [] | ||||||
|  | 
 | ||||||
|  |     workflows = ( | ||||||
|  |         Workflow.objects.filter(enabled=True, triggers__type=trigger_type) | ||||||
|  |         .prefetch_related( | ||||||
|  |             "actions", | ||||||
|  |             "actions__assign_view_users", | ||||||
|  |             "actions__assign_view_groups", | ||||||
|  |             "actions__assign_change_users", | ||||||
|  |             "actions__assign_change_groups", | ||||||
|  |             "actions__assign_custom_fields", | ||||||
|  |             "actions__remove_tags", | ||||||
|  |             "actions__remove_correspondents", | ||||||
|  |             "actions__remove_document_types", | ||||||
|  |             "actions__remove_storage_paths", | ||||||
|  |             "actions__remove_custom_fields", | ||||||
|  |             "actions__remove_owners", | ||||||
|  |             "triggers", | ||||||
|  |         ) | ||||||
|  |         .order_by("order") | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     for workflow in workflows: | ||||||
|  |         if not use_overrides: | ||||||
|             # This can be called from bulk_update_documents, which may be running multiple times |             # This can be called from bulk_update_documents, which may be running multiple times | ||||||
|             # Refresh this so the matching data is fresh and instance fields are re-freshed |             # Refresh this so the matching data is fresh and instance fields are re-freshed | ||||||
|             # Otherwise, this instance might be behind and overwrite the work another process did |             # Otherwise, this instance might be behind and overwrite the work another process did | ||||||
|             document.refresh_from_db() |             document.refresh_from_db() | ||||||
|         doc_tag_ids = list(document.tags.all().values_list("pk", flat=True)) |             doc_tag_ids = list(document.tags.values_list("pk", flat=True)) | ||||||
|         if matching.document_matches_workflow( | 
 | ||||||
|             document, |         if matching.document_matches_workflow(document, workflow, trigger_type): | ||||||
|             workflow, |  | ||||||
|             trigger_type, |  | ||||||
|         ): |  | ||||||
|             action: WorkflowAction |             action: WorkflowAction | ||||||
|             for action in workflow.actions.all(): |             for action in workflow.actions.all(): | ||||||
|                 logger.info( |                 message = f"Applying {action} from {workflow}" | ||||||
|                     f"Applying {action} from {workflow}", |                 if not use_overrides: | ||||||
|                     extra={"group": logging_group}, |                     logger.info(message, extra={"group": logging_group}) | ||||||
|                 ) |                 else: | ||||||
|  |                     messages.append(message) | ||||||
| 
 | 
 | ||||||
|                 if action.type == WorkflowAction.WorkflowActionType.ASSIGNMENT: |                 if action.type == WorkflowAction.WorkflowActionType.ASSIGNMENT: | ||||||
|                     assignment_action() |                     assignment_action() | ||||||
| 
 |  | ||||||
|                 elif action.type == WorkflowAction.WorkflowActionType.REMOVAL: |                 elif action.type == WorkflowAction.WorkflowActionType.REMOVAL: | ||||||
|                     removal_action() |                     removal_action() | ||||||
| 
 | 
 | ||||||
|  |             if not use_overrides: | ||||||
|                 # save first before setting tags |                 # save first before setting tags | ||||||
|                 document.save() |                 document.save() | ||||||
|                 document.tags.set(doc_tag_ids) |                 document.tags.set(doc_tag_ids) | ||||||
| 
 | 
 | ||||||
|  |     if use_overrides: | ||||||
|  |         return overrides, "\n".join(messages) | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| @before_task_publish.connect | @before_task_publish.connect | ||||||
| def before_task_publish_handler(sender=None, headers=None, body=None, **kwargs): | def before_task_publish_handler(sender=None, headers=None, body=None, **kwargs): | ||||||
|  | |||||||
							
								
								
									
										46
									
								
								src/documents/templating/title.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								src/documents/templating/title.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,46 @@ | |||||||
|  | from datetime import datetime | ||||||
|  | from pathlib import Path | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def parse_doc_title_w_placeholders( | ||||||
|  |     title: str, | ||||||
|  |     correspondent_name: str, | ||||||
|  |     doc_type_name: str, | ||||||
|  |     owner_username: str, | ||||||
|  |     local_added: datetime, | ||||||
|  |     original_filename: str, | ||||||
|  |     created: datetime | None = None, | ||||||
|  | ) -> str: | ||||||
|  |     """ | ||||||
|  |     Available title placeholders for Workflows depend on what has already been assigned, | ||||||
|  |     e.g. for pre-consumption triggers created will not have been parsed yet, but it will | ||||||
|  |     for added / updated triggers | ||||||
|  |     """ | ||||||
|  |     formatting = { | ||||||
|  |         "correspondent": correspondent_name, | ||||||
|  |         "document_type": doc_type_name, | ||||||
|  |         "added": local_added.isoformat(), | ||||||
|  |         "added_year": local_added.strftime("%Y"), | ||||||
|  |         "added_year_short": local_added.strftime("%y"), | ||||||
|  |         "added_month": local_added.strftime("%m"), | ||||||
|  |         "added_month_name": local_added.strftime("%B"), | ||||||
|  |         "added_month_name_short": local_added.strftime("%b"), | ||||||
|  |         "added_day": local_added.strftime("%d"), | ||||||
|  |         "added_time": local_added.strftime("%H:%M"), | ||||||
|  |         "owner_username": owner_username, | ||||||
|  |         "original_filename": Path(original_filename).stem, | ||||||
|  |     } | ||||||
|  |     if created is not None: | ||||||
|  |         formatting.update( | ||||||
|  |             { | ||||||
|  |                 "created": created.isoformat(), | ||||||
|  |                 "created_year": created.strftime("%Y"), | ||||||
|  |                 "created_year_short": created.strftime("%y"), | ||||||
|  |                 "created_month": created.strftime("%m"), | ||||||
|  |                 "created_month_name": created.strftime("%B"), | ||||||
|  |                 "created_month_name_short": created.strftime("%b"), | ||||||
|  |                 "created_day": created.strftime("%d"), | ||||||
|  |                 "created_time": created.strftime("%H:%M"), | ||||||
|  |             }, | ||||||
|  |         ) | ||||||
|  |     return title.format(**formatting).strip() | ||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user