mirror of
https://github.com/paperless-ngx/paperless-ngx.git
synced 2025-04-09 09:58:20 -05:00
Enhancement: title assignment placeholder error handling, fallback (#5282)
This commit is contained in:
parent
416ad13aaf
commit
6d5f4e92cc
@ -99,7 +99,7 @@
|
|||||||
<input type="hidden" formControlName="id" />
|
<input type="hidden" formControlName="id" />
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col">
|
<div class="col">
|
||||||
<pngx-input-text i18n-title title="Assign title" formControlName="assign_title" i18n-hint hint="Can include some placeholders, see <a target='_blank' href='https://docs.paperless-ngx.com/usage/#workflows'>documentation</a>." [error]="error?.assign_title"></pngx-input-text>
|
<pngx-input-text i18n-title title="Assign title" formControlName="assign_title" i18n-hint hint="Can include some placeholders, see <a target='_blank' href='https://docs.paperless-ngx.com/usage/#workflows'>documentation</a>." [error]="error?.actions?.[i]?.assign_title"></pngx-input-text>
|
||||||
<pngx-input-tags [allowCreate]="false" i18n-title title="Assign tags" formControlName="assign_tags"></pngx-input-tags>
|
<pngx-input-tags [allowCreate]="false" i18n-title title="Assign tags" formControlName="assign_tags"></pngx-input-tags>
|
||||||
<pngx-input-select i18n-title title="Assign document type" [items]="documentTypes" [allowNull]="true" formControlName="assign_document_type"></pngx-input-select>
|
<pngx-input-select i18n-title title="Assign document type" [items]="documentTypes" [allowNull]="true" formControlName="assign_document_type"></pngx-input-select>
|
||||||
<pngx-input-select i18n-title title="Assign correspondent" [items]="correspondents" [allowNull]="true" formControlName="assign_correspondent"></pngx-input-select>
|
<pngx-input-select i18n-title title="Assign correspondent" [items]="correspondents" [allowNull]="true" formControlName="assign_correspondent"></pngx-input-select>
|
||||||
|
@ -726,12 +726,17 @@ class Consumer(LoggingMixin):
|
|||||||
|
|
||||||
storage_type = Document.STORAGE_TYPE_UNENCRYPTED
|
storage_type = Document.STORAGE_TYPE_UNENCRYPTED
|
||||||
|
|
||||||
|
title = file_info.title[:127]
|
||||||
|
if self.override_title is not None:
|
||||||
|
try:
|
||||||
|
title = self._parse_title_placeholders(self.override_title)
|
||||||
|
except Exception as e:
|
||||||
|
self.log.error(
|
||||||
|
f"Error occurred parsing title override '{self.override_title}', falling back to original. Exception: {e}",
|
||||||
|
)
|
||||||
|
|
||||||
document = Document.objects.create(
|
document = Document.objects.create(
|
||||||
title=(
|
title=title,
|
||||||
self._parse_title_placeholders(self.override_title)
|
|
||||||
if self.override_title is not None
|
|
||||||
else file_info.title
|
|
||||||
)[:127],
|
|
||||||
content=text,
|
content=text,
|
||||||
mime_type=mime_type,
|
mime_type=mime_type,
|
||||||
checksum=hashlib.md5(self.working_copy.read_bytes()).hexdigest(),
|
checksum=hashlib.md5(self.working_copy.read_bytes()).hexdigest(),
|
||||||
|
@ -1386,12 +1386,38 @@ class WorkflowActionSerializer(serializers.ModelSerializer):
|
|||||||
|
|
||||||
def validate(self, attrs):
|
def validate(self, attrs):
|
||||||
# Empty strings treated as None to avoid unexpected behavior
|
# Empty strings treated as None to avoid unexpected behavior
|
||||||
if (
|
if "assign_title" in attrs:
|
||||||
"assign_title" in attrs
|
if attrs["assign_title"] is not None and len(attrs["assign_title"]) == 0:
|
||||||
and attrs["assign_title"] is not None
|
attrs["assign_title"] = None
|
||||||
and len(attrs["assign_title"]) == 0
|
else:
|
||||||
):
|
try:
|
||||||
attrs["assign_title"] = None
|
# test against all placeholders, see consumer.py `parse_doc_title_w_placeholders`
|
||||||
|
attrs["assign_title"].format(
|
||||||
|
correspondent="",
|
||||||
|
document_type="",
|
||||||
|
added="",
|
||||||
|
added_year="",
|
||||||
|
added_year_short="",
|
||||||
|
added_month="",
|
||||||
|
added_month_name="",
|
||||||
|
added_month_name_short="",
|
||||||
|
added_day="",
|
||||||
|
added_time="",
|
||||||
|
owner_username="",
|
||||||
|
original_filename="",
|
||||||
|
created="",
|
||||||
|
created_year="",
|
||||||
|
created_year_short="",
|
||||||
|
created_month="",
|
||||||
|
created_month_name="",
|
||||||
|
created_month_name_short="",
|
||||||
|
created_day="",
|
||||||
|
created_time="",
|
||||||
|
)
|
||||||
|
except (ValueError, KeyError) as e:
|
||||||
|
raise serializers.ValidationError(
|
||||||
|
{"assign_title": f'Invalid f-string detected: "{e.args[0]}"'},
|
||||||
|
)
|
||||||
|
|
||||||
return attrs
|
return attrs
|
||||||
|
|
||||||
|
@ -570,19 +570,27 @@ def run_workflow(
|
|||||||
document.owner = action.assign_owner
|
document.owner = action.assign_owner
|
||||||
|
|
||||||
if action.assign_title is not None:
|
if action.assign_title is not None:
|
||||||
document.title = parse_doc_title_w_placeholders(
|
try:
|
||||||
action.assign_title,
|
document.title = parse_doc_title_w_placeholders(
|
||||||
document.correspondent.name
|
action.assign_title,
|
||||||
if document.correspondent is not None
|
document.correspondent.name
|
||||||
else "",
|
if document.correspondent is not None
|
||||||
document.document_type.name
|
else "",
|
||||||
if document.document_type is not None
|
document.document_type.name
|
||||||
else "",
|
if document.document_type is not None
|
||||||
document.owner.username if document.owner is not None else "",
|
else "",
|
||||||
timezone.localtime(document.added),
|
document.owner.username
|
||||||
document.original_filename,
|
if document.owner is not None
|
||||||
timezone.localtime(document.created),
|
else "",
|
||||||
)
|
document.added,
|
||||||
|
document.original_filename,
|
||||||
|
document.created,
|
||||||
|
)
|
||||||
|
except Exception:
|
||||||
|
logger.exception(
|
||||||
|
f"Error occurred parsing title assignment '{action.assign_title}', falling back to original",
|
||||||
|
extra={"group": logging_group},
|
||||||
|
)
|
||||||
|
|
||||||
if (
|
if (
|
||||||
action.assign_view_users is not None
|
action.assign_view_users is not None
|
||||||
|
@ -248,6 +248,45 @@ class TestApiWorkflows(DirectoriesMixin, APITestCase):
|
|||||||
|
|
||||||
self.assertEqual(WorkflowTrigger.objects.count(), 1)
|
self.assertEqual(WorkflowTrigger.objects.count(), 1)
|
||||||
|
|
||||||
|
def test_api_create_invalid_assign_title(self):
|
||||||
|
"""
|
||||||
|
GIVEN:
|
||||||
|
- API request to create a workflow
|
||||||
|
- Invalid f-string for assign_title
|
||||||
|
WHEN:
|
||||||
|
- API is called
|
||||||
|
THEN:
|
||||||
|
- Correct HTTP 400 response
|
||||||
|
- No objects are created
|
||||||
|
"""
|
||||||
|
response = self.client.post(
|
||||||
|
self.ENDPOINT,
|
||||||
|
json.dumps(
|
||||||
|
{
|
||||||
|
"name": "Workflow 1",
|
||||||
|
"order": 1,
|
||||||
|
"triggers": [
|
||||||
|
{
|
||||||
|
"type": WorkflowTrigger.WorkflowTriggerType.DOCUMENT_UPDATED,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"actions": [
|
||||||
|
{
|
||||||
|
"assign_title": "{created_year]",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
),
|
||||||
|
content_type="application/json",
|
||||||
|
)
|
||||||
|
self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST)
|
||||||
|
self.assertIn(
|
||||||
|
"Invalid f-string detected",
|
||||||
|
response.data["actions"][0]["assign_title"][0],
|
||||||
|
)
|
||||||
|
|
||||||
|
self.assertEqual(Workflow.objects.count(), 1)
|
||||||
|
|
||||||
def test_api_create_workflow_trigger_action_empty_fields(self):
|
def test_api_create_workflow_trigger_action_empty_fields(self):
|
||||||
"""
|
"""
|
||||||
GIVEN:
|
GIVEN:
|
||||||
|
@ -423,6 +423,16 @@ class TestConsumer(DirectoriesMixin, FileSystemAssertsMixin, TestCase):
|
|||||||
self.assertEqual(document.title, "Override Title")
|
self.assertEqual(document.title, "Override Title")
|
||||||
self._assert_first_last_send_progress()
|
self._assert_first_last_send_progress()
|
||||||
|
|
||||||
|
def testOverrideTitleInvalidPlaceholders(self):
|
||||||
|
with self.assertLogs("paperless.consumer", level="ERROR") as cm:
|
||||||
|
document = self.consumer.try_consume_file(
|
||||||
|
self.get_test_file(),
|
||||||
|
override_title="Override {correspondent]",
|
||||||
|
)
|
||||||
|
self.assertEqual(document.title, "sample")
|
||||||
|
expected_str = "Error occurred parsing title override 'Override {correspondent]', falling back to original"
|
||||||
|
self.assertIn(expected_str, cm.output[0])
|
||||||
|
|
||||||
def testOverrideCorrespondent(self):
|
def testOverrideCorrespondent(self):
|
||||||
c = Correspondent.objects.create(name="test")
|
c = Correspondent.objects.create(name="test")
|
||||||
|
|
||||||
|
@ -966,6 +966,50 @@ class TestWorkflows(DirectoriesMixin, FileSystemAssertsMixin, APITestCase):
|
|||||||
expected_str = f"Document correspondent {doc.correspondent} does not match {trigger.filter_has_correspondent}"
|
expected_str = f"Document correspondent {doc.correspondent} does not match {trigger.filter_has_correspondent}"
|
||||||
self.assertIn(expected_str, cm.output[1])
|
self.assertIn(expected_str, cm.output[1])
|
||||||
|
|
||||||
|
def test_document_added_invalid_title_placeholders(self):
|
||||||
|
"""
|
||||||
|
GIVEN:
|
||||||
|
- Existing workflow with added trigger type
|
||||||
|
- Assign title field has an error
|
||||||
|
WHEN:
|
||||||
|
- File that matches is added
|
||||||
|
THEN:
|
||||||
|
- Title is not updated, error is output
|
||||||
|
"""
|
||||||
|
trigger = WorkflowTrigger.objects.create(
|
||||||
|
type=WorkflowTrigger.WorkflowTriggerType.DOCUMENT_ADDED,
|
||||||
|
filter_filename="*sample*",
|
||||||
|
)
|
||||||
|
action = WorkflowAction.objects.create(
|
||||||
|
assign_title="Doc {created_year]",
|
||||||
|
)
|
||||||
|
w = Workflow.objects.create(
|
||||||
|
name="Workflow 1",
|
||||||
|
order=0,
|
||||||
|
)
|
||||||
|
w.triggers.add(trigger)
|
||||||
|
w.actions.add(action)
|
||||||
|
w.save()
|
||||||
|
|
||||||
|
now = timezone.localtime(timezone.now())
|
||||||
|
created = now - timedelta(weeks=520)
|
||||||
|
doc = Document.objects.create(
|
||||||
|
original_filename="sample.pdf",
|
||||||
|
title="sample test",
|
||||||
|
content="Hello world bar",
|
||||||
|
created=created,
|
||||||
|
)
|
||||||
|
|
||||||
|
with self.assertLogs("paperless.handlers", level="ERROR") as cm:
|
||||||
|
document_consumption_finished.send(
|
||||||
|
sender=self.__class__,
|
||||||
|
document=doc,
|
||||||
|
)
|
||||||
|
expected_str = f"Error occurred parsing title assignment '{action.assign_title}', falling back to original"
|
||||||
|
self.assertIn(expected_str, cm.output[0])
|
||||||
|
|
||||||
|
self.assertEqual(doc.title, "sample test")
|
||||||
|
|
||||||
def test_document_updated_workflow(self):
|
def test_document_updated_workflow(self):
|
||||||
trigger = WorkflowTrigger.objects.create(
|
trigger = WorkflowTrigger.objects.create(
|
||||||
type=WorkflowTrigger.WorkflowTriggerType.DOCUMENT_UPDATED,
|
type=WorkflowTrigger.WorkflowTriggerType.DOCUMENT_UPDATED,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user