import datetime import logging import os import re from collections import OrderedDict from pathlib import Path from typing import Final import dateutil.parser import pathvalidate from celery import states from django.conf import settings from django.contrib.auth.models import Group from django.contrib.auth.models import User from django.core.validators import MaxValueValidator from django.core.validators import MinValueValidator from django.db import models from django.utils import timezone from django.utils.translation import gettext_lazy as _ from multiselectfield import MultiSelectField if settings.AUDIT_LOG_ENABLED: from auditlog.registry import auditlog from django.db.models import Case from django.db.models.functions import Cast from django.db.models.functions import Substr from django_softdelete.models import SoftDeleteModel from documents.data_models import DocumentSource from documents.parsers import get_default_file_extension class ModelWithOwner(models.Model): owner = models.ForeignKey( User, blank=True, null=True, default=None, on_delete=models.SET_NULL, verbose_name=_("owner"), ) class Meta: abstract = True class MatchingModel(ModelWithOwner): MATCH_NONE = 0 MATCH_ANY = 1 MATCH_ALL = 2 MATCH_LITERAL = 3 MATCH_REGEX = 4 MATCH_FUZZY = 5 MATCH_AUTO = 6 MATCHING_ALGORITHMS = ( (MATCH_NONE, _("None")), (MATCH_ANY, _("Any word")), (MATCH_ALL, _("All words")), (MATCH_LITERAL, _("Exact match")), (MATCH_REGEX, _("Regular expression")), (MATCH_FUZZY, _("Fuzzy word")), (MATCH_AUTO, _("Automatic")), ) name = models.CharField(_("name"), max_length=128) match = models.CharField(_("match"), max_length=256, blank=True) matching_algorithm = models.PositiveIntegerField( _("matching algorithm"), choices=MATCHING_ALGORITHMS, default=MATCH_ANY, ) is_insensitive = models.BooleanField(_("is insensitive"), default=True) class Meta: abstract = True ordering = ("name",) constraints = [ models.UniqueConstraint( fields=["name", "owner"], name="%(app_label)s_%(class)s_unique_name_owner", ), models.UniqueConstraint( name="%(app_label)s_%(class)s_name_uniq", fields=["name"], condition=models.Q(owner__isnull=True), ), ] def __str__(self): return self.name class Correspondent(MatchingModel): class Meta(MatchingModel.Meta): verbose_name = _("correspondent") verbose_name_plural = _("correspondents") class Tag(MatchingModel): color = models.CharField(_("color"), max_length=7, default="#a6cee3") is_inbox_tag = models.BooleanField( _("is inbox tag"), default=False, help_text=_( "Marks this tag as an inbox tag: All newly consumed " "documents will be tagged with inbox tags.", ), ) class Meta(MatchingModel.Meta): verbose_name = _("tag") verbose_name_plural = _("tags") class DocumentType(MatchingModel): class Meta(MatchingModel.Meta): verbose_name = _("document type") verbose_name_plural = _("document types") class StoragePath(MatchingModel): path = models.TextField( _("path"), ) class Meta(MatchingModel.Meta): verbose_name = _("storage path") verbose_name_plural = _("storage paths") class Document(SoftDeleteModel, ModelWithOwner): STORAGE_TYPE_UNENCRYPTED = "unencrypted" STORAGE_TYPE_GPG = "gpg" STORAGE_TYPES = ( (STORAGE_TYPE_UNENCRYPTED, _("Unencrypted")), (STORAGE_TYPE_GPG, _("Encrypted with GNU Privacy Guard")), ) correspondent = models.ForeignKey( Correspondent, blank=True, null=True, related_name="documents", on_delete=models.SET_NULL, verbose_name=_("correspondent"), ) storage_path = models.ForeignKey( StoragePath, blank=True, null=True, related_name="documents", on_delete=models.SET_NULL, verbose_name=_("storage path"), ) title = models.CharField(_("title"), max_length=128, blank=True, db_index=True) document_type = models.ForeignKey( DocumentType, blank=True, null=True, related_name="documents", on_delete=models.SET_NULL, verbose_name=_("document type"), ) content = models.TextField( _("content"), blank=True, help_text=_( "The raw, text-only data of the document. This field is " "primarily used for searching.", ), ) mime_type = models.CharField(_("mime type"), max_length=256, editable=False) tags = models.ManyToManyField( Tag, related_name="documents", blank=True, verbose_name=_("tags"), ) checksum = models.CharField( _("checksum"), max_length=32, editable=False, unique=True, help_text=_("The checksum of the original document."), ) archive_checksum = models.CharField( _("archive checksum"), max_length=32, editable=False, blank=True, null=True, help_text=_("The checksum of the archived document."), ) page_count = models.PositiveIntegerField( _("page count"), blank=False, null=True, unique=False, db_index=False, validators=[MinValueValidator(1)], help_text=_( "The number of pages of the document.", ), ) created = models.DateTimeField(_("created"), default=timezone.now, db_index=True) modified = models.DateTimeField( _("modified"), auto_now=True, editable=False, db_index=True, ) storage_type = models.CharField( _("storage type"), max_length=11, choices=STORAGE_TYPES, default=STORAGE_TYPE_UNENCRYPTED, editable=False, ) added = models.DateTimeField( _("added"), default=timezone.now, editable=False, db_index=True, ) filename = models.FilePathField( _("filename"), max_length=1024, editable=False, default=None, unique=True, null=True, help_text=_("Current filename in storage"), ) archive_filename = models.FilePathField( _("archive filename"), max_length=1024, editable=False, default=None, unique=True, null=True, help_text=_("Current archive filename in storage"), ) original_filename = models.CharField( _("original filename"), max_length=1024, editable=False, default=None, unique=False, null=True, help_text=_("The original name of the file when it was uploaded"), ) ARCHIVE_SERIAL_NUMBER_MIN: Final[int] = 0 ARCHIVE_SERIAL_NUMBER_MAX: Final[int] = 0xFF_FF_FF_FF archive_serial_number = models.PositiveIntegerField( _("archive serial number"), blank=True, null=True, unique=True, db_index=True, validators=[ MaxValueValidator(ARCHIVE_SERIAL_NUMBER_MAX), MinValueValidator(ARCHIVE_SERIAL_NUMBER_MIN), ], help_text=_( "The position of this document in your physical document archive.", ), ) class Meta: ordering = ("-created",) verbose_name = _("document") verbose_name_plural = _("documents") def __str__(self) -> str: # Convert UTC database time to local time created = datetime.date.isoformat(timezone.localdate(self.created)) res = f"{created}" if self.correspondent: res += f" {self.correspondent}" if self.title: res += f" {self.title}" return res @property def source_path(self) -> Path: if self.filename: fname = str(self.filename) else: fname = f"{self.pk:07}{self.file_type}" if self.storage_type == self.STORAGE_TYPE_GPG: fname += ".gpg" # pragma: no cover return (settings.ORIGINALS_DIR / Path(fname)).resolve() @property def source_file(self): return open(self.source_path, "rb") @property def has_archive_version(self) -> bool: return self.archive_filename is not None @property def archive_path(self) -> Path | None: if self.has_archive_version: return (settings.ARCHIVE_DIR / Path(str(self.archive_filename))).resolve() else: return None @property def archive_file(self): return open(self.archive_path, "rb") def get_public_filename(self, archive=False, counter=0, suffix=None) -> str: """ Returns a sanitized filename for the document, not including any paths. """ result = str(self) if counter: result += f"_{counter:02}" if suffix: result += suffix if archive: result += ".pdf" else: result += self.file_type return pathvalidate.sanitize_filename(result, replacement_text="-") @property def file_type(self): return get_default_file_extension(self.mime_type) @property def thumbnail_path(self) -> Path: webp_file_name = f"{self.pk:07}.webp" if self.storage_type == self.STORAGE_TYPE_GPG: webp_file_name += ".gpg" webp_file_path = settings.THUMBNAIL_DIR / Path(webp_file_name) return webp_file_path.resolve() @property def thumbnail_file(self): return open(self.thumbnail_path, "rb") @property def created_date(self): return timezone.localdate(self.created) class Log(models.Model): LEVELS = ( (logging.DEBUG, _("debug")), (logging.INFO, _("information")), (logging.WARNING, _("warning")), (logging.ERROR, _("error")), (logging.CRITICAL, _("critical")), ) group = models.UUIDField(_("group"), blank=True, null=True) message = models.TextField(_("message")) level = models.PositiveIntegerField( _("level"), choices=LEVELS, default=logging.INFO, ) created = models.DateTimeField(_("created"), auto_now_add=True) class Meta: ordering = ("-created",) verbose_name = _("log") verbose_name_plural = _("logs") def __str__(self): return self.message class SavedView(ModelWithOwner): class DisplayMode(models.TextChoices): TABLE = ("table", _("Table")) SMALL_CARDS = ("smallCards", _("Small Cards")) LARGE_CARDS = ("largeCards", _("Large Cards")) class DisplayFields(models.TextChoices): TITLE = ("title", _("Title")) CREATED = ("created", _("Created")) ADDED = ("added", _("Added")) TAGS = ("tag"), _("Tags") CORRESPONDENT = ("correspondent", _("Correspondent")) DOCUMENT_TYPE = ("documenttype", _("Document Type")) STORAGE_PATH = ("storagepath", _("Storage Path")) NOTES = ("note", _("Note")) OWNER = ("owner", _("Owner")) SHARED = ("shared", _("Shared")) ASN = ("asn", _("ASN")) PAGE_COUNT = ("pagecount", _("Pages")) CUSTOM_FIELD = ("custom_field_%d", ("Custom Field")) name = models.CharField(_("name"), max_length=128) show_on_dashboard = models.BooleanField( _("show on dashboard"), ) show_in_sidebar = models.BooleanField( _("show in sidebar"), ) sort_field = models.CharField( _("sort field"), max_length=128, null=True, blank=True, ) sort_reverse = models.BooleanField(_("sort reverse"), default=False) page_size = models.PositiveIntegerField( _("View page size"), null=True, blank=True, validators=[MinValueValidator(1)], ) display_mode = models.CharField( max_length=128, verbose_name=_("View display mode"), choices=DisplayMode.choices, null=True, blank=True, ) display_fields = models.JSONField( verbose_name=_("Document display fields"), null=True, blank=True, ) class Meta: ordering = ("name",) verbose_name = _("saved view") verbose_name_plural = _("saved views") def __str__(self): return f"SavedView {self.name}" class SavedViewFilterRule(models.Model): RULE_TYPES = [ (0, _("title contains")), (1, _("content contains")), (2, _("ASN is")), (3, _("correspondent is")), (4, _("document type is")), (5, _("is in inbox")), (6, _("has tag")), (7, _("has any tag")), (8, _("created before")), (9, _("created after")), (10, _("created year is")), (11, _("created month is")), (12, _("created day is")), (13, _("added before")), (14, _("added after")), (15, _("modified before")), (16, _("modified after")), (17, _("does not have tag")), (18, _("does not have ASN")), (19, _("title or content contains")), (20, _("fulltext query")), (21, _("more like this")), (22, _("has tags in")), (23, _("ASN greater than")), (24, _("ASN less than")), (25, _("storage path is")), (26, _("has correspondent in")), (27, _("does not have correspondent in")), (28, _("has document type in")), (29, _("does not have document type in")), (30, _("has storage path in")), (31, _("does not have storage path in")), (32, _("owner is")), (33, _("has owner in")), (34, _("does not have owner")), (35, _("does not have owner in")), (36, _("has custom field value")), (37, _("is shared by me")), (38, _("has custom fields")), (39, _("has custom field in")), (40, _("does not have custom field in")), (41, _("does not have custom field")), (42, _("custom fields query")), ] saved_view = models.ForeignKey( SavedView, on_delete=models.CASCADE, related_name="filter_rules", verbose_name=_("saved view"), ) rule_type = models.PositiveIntegerField(_("rule type"), choices=RULE_TYPES) value = models.CharField(_("value"), max_length=255, blank=True, null=True) class Meta: verbose_name = _("filter rule") verbose_name_plural = _("filter rules") def __str__(self) -> str: return f"SavedViewFilterRule: {self.rule_type} : {self.value}" # TODO: why is this in the models file? # TODO: how about, what is this and where is it documented? # It appears to parsing JSON from an environment variable to get a title and date from # the filename, if possible, as a higher priority than either document filename or # content parsing class FileInfo: REGEXES = OrderedDict( [ ( "created-title", re.compile( r"^(?P\d{8}(\d{6})?Z) - (?P.*)$", flags=re.IGNORECASE, ), ), ("title", re.compile(r"(?P<title>.*)$", flags=re.IGNORECASE)), ], ) def __init__( self, created=None, correspondent=None, title=None, tags=(), extension=None, ): self.created = created self.title = title self.extension = extension self.correspondent = correspondent self.tags = tags @classmethod def _get_created(cls, created): try: return dateutil.parser.parse(f"{created[:-1]:0<14}Z") except ValueError: return None @classmethod def _get_title(cls, title): return title @classmethod def _mangle_property(cls, properties, name): if name in properties: properties[name] = getattr(cls, f"_get_{name}")(properties[name]) @classmethod def from_filename(cls, filename) -> "FileInfo": # Mutate filename in-place before parsing its components # by applying at most one of the configured transformations. for pattern, repl in settings.FILENAME_PARSE_TRANSFORMS: (filename, count) = pattern.subn(repl, filename) if count: break # do this after the transforms so that the transforms can do whatever # with the file extension. filename_no_ext = os.path.splitext(filename)[0] if filename_no_ext == filename and filename.startswith("."): # This is a very special case where there is no text before the # file type. # TODO: this should be handled better. The ext is not removed # because usually, files like '.pdf' are just hidden files # with the name pdf, but in our case, its more likely that # there's just no name to begin with. filename = "" # This isn't too bad either, since we'll just not match anything # and return an empty title. TODO: actually, this is kinda bad. else: filename = filename_no_ext # Parse filename components. for regex in cls.REGEXES.values(): m = regex.match(filename) if m: properties = m.groupdict() cls._mangle_property(properties, "created") cls._mangle_property(properties, "title") return cls(**properties) # Extending User Model Using a One-To-One Link class UiSettings(models.Model): user = models.OneToOneField( User, on_delete=models.CASCADE, related_name="ui_settings", ) settings = models.JSONField(null=True) def __str__(self): return self.user.username class PaperlessTask(ModelWithOwner): ALL_STATES = sorted(states.ALL_STATES) TASK_STATE_CHOICES = sorted(zip(ALL_STATES, ALL_STATES)) task_id = models.CharField( max_length=255, unique=True, verbose_name=_("Task ID"), help_text=_("Celery ID for the Task that was run"), ) acknowledged = models.BooleanField( default=False, verbose_name=_("Acknowledged"), help_text=_("If the task is acknowledged via the frontend or API"), ) task_file_name = models.CharField( null=True, max_length=255, verbose_name=_("Task Filename"), help_text=_("Name of the file which the Task was run for"), ) task_name = models.CharField( null=True, max_length=255, verbose_name=_("Task Name"), help_text=_("Name of the Task which was run"), ) status = models.CharField( max_length=30, default=states.PENDING, choices=TASK_STATE_CHOICES, verbose_name=_("Task State"), help_text=_("Current state of the task being run"), ) date_created = models.DateTimeField( null=True, default=timezone.now, verbose_name=_("Created DateTime"), help_text=_("Datetime field when the task result was created in UTC"), ) date_started = models.DateTimeField( null=True, default=None, verbose_name=_("Started DateTime"), help_text=_("Datetime field when the task was started in UTC"), ) date_done = models.DateTimeField( null=True, default=None, verbose_name=_("Completed DateTime"), help_text=_("Datetime field when the task was completed in UTC"), ) result = models.TextField( null=True, default=None, verbose_name=_("Result Data"), help_text=_( "The data returned by the task", ), ) def __str__(self) -> str: return f"Task {self.task_id}" class Note(SoftDeleteModel): note = models.TextField( _("content"), blank=True, help_text=_("Note for the document"), ) created = models.DateTimeField( _("created"), default=timezone.now, db_index=True, ) document = models.ForeignKey( Document, blank=True, null=True, related_name="notes", on_delete=models.CASCADE, verbose_name=_("document"), ) user = models.ForeignKey( User, blank=True, null=True, related_name="notes", on_delete=models.SET_NULL, verbose_name=_("user"), ) class Meta: ordering = ("created",) verbose_name = _("note") verbose_name_plural = _("notes") def __str__(self): return self.note class ShareLink(SoftDeleteModel): class FileVersion(models.TextChoices): ARCHIVE = ("archive", _("Archive")) ORIGINAL = ("original", _("Original")) created = models.DateTimeField( _("created"), default=timezone.now, db_index=True, blank=True, editable=False, ) expiration = models.DateTimeField( _("expiration"), blank=True, null=True, db_index=True, ) slug = models.SlugField( _("slug"), db_index=True, unique=True, blank=True, editable=False, ) document = models.ForeignKey( Document, blank=True, related_name="share_links", on_delete=models.CASCADE, verbose_name=_("document"), ) file_version = models.CharField( max_length=50, choices=FileVersion.choices, default=FileVersion.ARCHIVE, ) owner = models.ForeignKey( User, blank=True, null=True, related_name="share_links", on_delete=models.SET_NULL, verbose_name=_("owner"), ) class Meta: ordering = ("created",) verbose_name = _("share link") verbose_name_plural = _("share links") def __str__(self): return f"Share Link for {self.document.title}" class CustomField(models.Model): """ Defines the name and type of a custom field """ class FieldDataType(models.TextChoices): STRING = ("string", _("String")) URL = ("url", _("URL")) DATE = ("date", _("Date")) BOOL = ("boolean"), _("Boolean") INT = ("integer", _("Integer")) FLOAT = ("float", _("Float")) MONETARY = ("monetary", _("Monetary")) DOCUMENTLINK = ("documentlink", _("Document Link")) SELECT = ("select", _("Select")) created = models.DateTimeField( _("created"), default=timezone.now, db_index=True, editable=False, ) name = models.CharField(max_length=128) data_type = models.CharField( _("data type"), max_length=50, choices=FieldDataType.choices, editable=False, ) extra_data = models.JSONField( _("extra data"), null=True, blank=True, help_text=_( "Extra data for the custom field, such as select options", ), ) class Meta: ordering = ("created",) verbose_name = _("custom field") verbose_name_plural = _("custom fields") constraints = [ models.UniqueConstraint( fields=["name"], name="%(app_label)s_%(class)s_unique_name", ), ] def __str__(self) -> str: return f"{self.name} : {self.data_type}" class CustomFieldInstance(SoftDeleteModel): """ A single instance of a field, attached to a CustomField for the name and type and attached to a single Document to be metadata for it """ TYPE_TO_DATA_STORE_NAME_MAP = { CustomField.FieldDataType.STRING: "value_text", CustomField.FieldDataType.URL: "value_url", CustomField.FieldDataType.DATE: "value_date", CustomField.FieldDataType.BOOL: "value_bool", CustomField.FieldDataType.INT: "value_int", CustomField.FieldDataType.FLOAT: "value_float", CustomField.FieldDataType.MONETARY: "value_monetary", CustomField.FieldDataType.DOCUMENTLINK: "value_document_ids", CustomField.FieldDataType.SELECT: "value_select", } created = models.DateTimeField( _("created"), default=timezone.now, db_index=True, editable=False, ) document = models.ForeignKey( Document, blank=False, null=False, on_delete=models.CASCADE, related_name="custom_fields", editable=False, ) field = models.ForeignKey( CustomField, blank=False, null=False, on_delete=models.CASCADE, related_name="fields", editable=False, ) # Actual data storage value_text = models.CharField(max_length=128, null=True) value_bool = models.BooleanField(null=True) value_url = models.URLField(null=True) value_date = models.DateField(null=True) value_int = models.IntegerField(null=True) value_float = models.FloatField(null=True) value_monetary = models.CharField(null=True, max_length=128) value_monetary_amount = models.GeneratedField( expression=Case( # If the value starts with a number and no currency symbol, use the whole string models.When( value_monetary__regex=r"^\d+", then=Cast( Substr("value_monetary", 1), output_field=models.DecimalField(decimal_places=2, max_digits=65), ), ), # If the value starts with a 3-char currency symbol, use the rest of the string default=Cast( Substr("value_monetary", 4), output_field=models.DecimalField(decimal_places=2, max_digits=65), ), output_field=models.DecimalField(decimal_places=2, max_digits=65), ), output_field=models.DecimalField(decimal_places=2, max_digits=65), db_persist=True, ) value_document_ids = models.JSONField(null=True) value_select = models.PositiveSmallIntegerField(null=True) class Meta: ordering = ("created",) verbose_name = _("custom field instance") verbose_name_plural = _("custom field instances") constraints = [ models.UniqueConstraint( fields=["document", "field"], name="%(app_label)s_%(class)s_unique_document_field", ), ] def __str__(self) -> str: value = ( self.field.extra_data["select_options"][self.value_select] if ( self.field.data_type == CustomField.FieldDataType.SELECT and self.value_select is not None ) else self.value ) return str(self.field.name) + f" : {value}" @classmethod def get_value_field_name(cls, data_type: CustomField.FieldDataType): try: return cls.TYPE_TO_DATA_STORE_NAME_MAP[data_type] except KeyError: # pragma: no cover raise NotImplementedError(data_type) @property def value(self): """ Based on the data type, access the actual value the instance stores A little shorthand/quick way to get what is actually here """ value_field_name = self.get_value_field_name(self.field.data_type) return getattr(self, value_field_name) if settings.AUDIT_LOG_ENABLED: auditlog.register( Document, m2m_fields={"tags"}, exclude_fields=["modified"], ) auditlog.register(Correspondent) auditlog.register(Tag) auditlog.register(DocumentType) auditlog.register(Note) auditlog.register(CustomField) auditlog.register(CustomFieldInstance) class WorkflowTrigger(models.Model): class WorkflowTriggerMatching(models.IntegerChoices): # No auto matching NONE = MatchingModel.MATCH_NONE, _("None") ANY = MatchingModel.MATCH_ANY, _("Any word") ALL = MatchingModel.MATCH_ALL, _("All words") LITERAL = MatchingModel.MATCH_LITERAL, _("Exact match") REGEX = MatchingModel.MATCH_REGEX, _("Regular expression") FUZZY = MatchingModel.MATCH_FUZZY, _("Fuzzy word") class WorkflowTriggerType(models.IntegerChoices): CONSUMPTION = 1, _("Consumption Started") DOCUMENT_ADDED = 2, _("Document Added") DOCUMENT_UPDATED = 3, _("Document Updated") class DocumentSourceChoices(models.IntegerChoices): CONSUME_FOLDER = DocumentSource.ConsumeFolder.value, _("Consume Folder") API_UPLOAD = DocumentSource.ApiUpload.value, _("Api Upload") MAIL_FETCH = DocumentSource.MailFetch.value, _("Mail Fetch") type = models.PositiveIntegerField( _("Workflow Trigger Type"), choices=WorkflowTriggerType.choices, default=WorkflowTriggerType.CONSUMPTION, ) sources = MultiSelectField( max_length=5, choices=DocumentSourceChoices.choices, default=f"{DocumentSource.ConsumeFolder},{DocumentSource.ApiUpload},{DocumentSource.MailFetch}", ) filter_path = models.CharField( _("filter path"), max_length=256, null=True, blank=True, help_text=_( "Only consume documents with a path that matches " "this if specified. Wildcards specified as * are " "allowed. Case insensitive.", ), ) filter_filename = models.CharField( _("filter filename"), max_length=256, null=True, blank=True, help_text=_( "Only consume documents which entirely match this " "filename if specified. Wildcards such as *.pdf or " "*invoice* are allowed. Case insensitive.", ), ) filter_mailrule = models.ForeignKey( "paperless_mail.MailRule", null=True, blank=True, on_delete=models.SET_NULL, verbose_name=_("filter documents from this mail rule"), ) match = models.CharField(_("match"), max_length=256, blank=True) matching_algorithm = models.PositiveIntegerField( _("matching algorithm"), choices=WorkflowTriggerMatching.choices, default=WorkflowTriggerMatching.NONE, ) is_insensitive = models.BooleanField(_("is insensitive"), default=True) filter_has_tags = models.ManyToManyField( Tag, blank=True, verbose_name=_("has these tag(s)"), ) filter_has_document_type = models.ForeignKey( DocumentType, null=True, blank=True, on_delete=models.SET_NULL, verbose_name=_("has this document type"), ) filter_has_correspondent = models.ForeignKey( Correspondent, null=True, blank=True, on_delete=models.SET_NULL, verbose_name=_("has this correspondent"), ) class Meta: verbose_name = _("workflow trigger") verbose_name_plural = _("workflow triggers") def __str__(self): return f"WorkflowTrigger {self.pk}" class WorkflowAction(models.Model): class WorkflowActionType(models.IntegerChoices): ASSIGNMENT = ( 1, _("Assignment"), ) REMOVAL = ( 2, _("Removal"), ) type = models.PositiveIntegerField( _("Workflow Action Type"), choices=WorkflowActionType.choices, default=WorkflowActionType.ASSIGNMENT, ) assign_title = models.CharField( _("assign title"), max_length=256, null=True, blank=True, help_text=_( "Assign a document title, can include some placeholders, " "see documentation.", ), ) assign_tags = models.ManyToManyField( Tag, blank=True, related_name="+", verbose_name=_("assign this tag"), ) assign_document_type = models.ForeignKey( DocumentType, null=True, blank=True, on_delete=models.SET_NULL, related_name="+", verbose_name=_("assign this document type"), ) assign_correspondent = models.ForeignKey( Correspondent, null=True, blank=True, on_delete=models.SET_NULL, related_name="+", verbose_name=_("assign this correspondent"), ) assign_storage_path = models.ForeignKey( StoragePath, null=True, blank=True, on_delete=models.SET_NULL, related_name="+", verbose_name=_("assign this storage path"), ) assign_owner = models.ForeignKey( User, null=True, blank=True, on_delete=models.SET_NULL, related_name="+", verbose_name=_("assign this owner"), ) assign_view_users = models.ManyToManyField( User, blank=True, related_name="+", verbose_name=_("grant view permissions to these users"), ) assign_view_groups = models.ManyToManyField( Group, blank=True, related_name="+", verbose_name=_("grant view permissions to these groups"), ) assign_change_users = models.ManyToManyField( User, blank=True, related_name="+", verbose_name=_("grant change permissions to these users"), ) assign_change_groups = models.ManyToManyField( Group, blank=True, related_name="+", verbose_name=_("grant change permissions to these groups"), ) assign_custom_fields = models.ManyToManyField( CustomField, blank=True, related_name="+", verbose_name=_("assign these custom fields"), ) remove_tags = models.ManyToManyField( Tag, blank=True, related_name="+", verbose_name=_("remove these tag(s)"), ) remove_all_tags = models.BooleanField( default=False, verbose_name=_("remove all tags"), ) remove_document_types = models.ManyToManyField( DocumentType, blank=True, related_name="+", verbose_name=_("remove these document type(s)"), ) remove_all_document_types = models.BooleanField( default=False, verbose_name=_("remove all document types"), ) remove_correspondents = models.ManyToManyField( Correspondent, blank=True, related_name="+", verbose_name=_("remove these correspondent(s)"), ) remove_all_correspondents = models.BooleanField( default=False, verbose_name=_("remove all correspondents"), ) remove_storage_paths = models.ManyToManyField( StoragePath, blank=True, related_name="+", verbose_name=_("remove these storage path(s)"), ) remove_all_storage_paths = models.BooleanField( default=False, verbose_name=_("remove all storage paths"), ) remove_owners = models.ManyToManyField( User, blank=True, related_name="+", verbose_name=_("remove these owner(s)"), ) remove_all_owners = models.BooleanField( default=False, verbose_name=_("remove all owners"), ) remove_view_users = models.ManyToManyField( User, blank=True, related_name="+", verbose_name=_("remove view permissions for these users"), ) remove_view_groups = models.ManyToManyField( Group, blank=True, related_name="+", verbose_name=_("remove view permissions for these groups"), ) remove_change_users = models.ManyToManyField( User, blank=True, related_name="+", verbose_name=_("remove change permissions for these users"), ) remove_change_groups = models.ManyToManyField( Group, blank=True, related_name="+", verbose_name=_("remove change permissions for these groups"), ) remove_all_permissions = models.BooleanField( default=False, verbose_name=_("remove all permissions"), ) remove_custom_fields = models.ManyToManyField( CustomField, blank=True, related_name="+", verbose_name=_("remove these custom fields"), ) remove_all_custom_fields = models.BooleanField( default=False, verbose_name=_("remove all custom fields"), ) class Meta: verbose_name = _("workflow action") verbose_name_plural = _("workflow actions") def __str__(self): return f"WorkflowAction {self.pk}" class Workflow(models.Model): name = models.CharField(_("name"), max_length=256, unique=True) order = models.IntegerField(_("order"), default=0) triggers = models.ManyToManyField( WorkflowTrigger, related_name="workflows", blank=False, verbose_name=_("triggers"), ) actions = models.ManyToManyField( WorkflowAction, related_name="workflows", blank=False, verbose_name=_("actions"), ) enabled = models.BooleanField(_("enabled"), default=True) def __str__(self): return f"Workflow: {self.name}"