mirror of
https://github.com/paperless-ngx/paperless-ngx.git
synced 2025-04-02 13:45:10 -05:00
304 lines
11 KiB
Python
304 lines
11 KiB
Python
import hashlib
|
|
import logging
|
|
import shutil
|
|
import uuid
|
|
from typing import Optional
|
|
from typing import Type
|
|
|
|
import tqdm
|
|
from asgiref.sync import async_to_sync
|
|
from celery import shared_task
|
|
from channels.layers import get_channel_layer
|
|
from django.conf import settings
|
|
from django.db import transaction
|
|
from django.db.models.signals import post_save
|
|
from documents import barcodes
|
|
from documents import index
|
|
from documents import sanity_checker
|
|
from documents.classifier import DocumentClassifier
|
|
from documents.classifier import load_classifier
|
|
from documents.consumer import Consumer
|
|
from documents.consumer import ConsumerError
|
|
from documents.data_models import ConsumableDocument
|
|
from documents.data_models import DocumentMetadataOverrides
|
|
from documents.data_models import DocumentSource
|
|
from documents.file_handling import create_source_path_directory
|
|
from documents.file_handling import generate_unique_filename
|
|
from documents.models import Correspondent
|
|
from documents.models import Document
|
|
from documents.models import DocumentType
|
|
from documents.models import StoragePath
|
|
from documents.models import Tag
|
|
from documents.parsers import DocumentParser
|
|
from documents.parsers import get_parser_class_for_mime_type
|
|
from documents.sanity_checker import SanityCheckFailedException
|
|
from filelock import FileLock
|
|
from redis.exceptions import ConnectionError
|
|
from whoosh.writing import AsyncWriter
|
|
|
|
|
|
logger = logging.getLogger("paperless.tasks")
|
|
|
|
|
|
@shared_task
|
|
def index_optimize():
|
|
ix = index.open_index()
|
|
writer = AsyncWriter(ix)
|
|
writer.commit(optimize=True)
|
|
|
|
|
|
def index_reindex(progress_bar_disable=False):
|
|
documents = Document.objects.all()
|
|
|
|
ix = index.open_index(recreate=True)
|
|
|
|
with AsyncWriter(ix) as writer:
|
|
for document in tqdm.tqdm(documents, disable=progress_bar_disable):
|
|
index.update_document(writer, document)
|
|
|
|
|
|
@shared_task
|
|
def train_classifier():
|
|
if (
|
|
not Tag.objects.filter(matching_algorithm=Tag.MATCH_AUTO).exists()
|
|
and not DocumentType.objects.filter(matching_algorithm=Tag.MATCH_AUTO).exists()
|
|
and not Correspondent.objects.filter(matching_algorithm=Tag.MATCH_AUTO).exists()
|
|
and not StoragePath.objects.filter(matching_algorithm=Tag.MATCH_AUTO).exists()
|
|
):
|
|
|
|
return
|
|
|
|
classifier = load_classifier()
|
|
|
|
if not classifier:
|
|
classifier = DocumentClassifier()
|
|
|
|
try:
|
|
if classifier.train():
|
|
logger.info(
|
|
f"Saving updated classifier model to {settings.MODEL_FILE}...",
|
|
)
|
|
classifier.save()
|
|
else:
|
|
logger.debug("Training data unchanged.")
|
|
|
|
except Exception as e:
|
|
logger.warning("Classifier error: " + str(e))
|
|
|
|
|
|
@shared_task
|
|
def consume_file(
|
|
input_doc: ConsumableDocument,
|
|
overrides: Optional[DocumentMetadataOverrides] = None,
|
|
):
|
|
|
|
# Default no overrides
|
|
if overrides is None:
|
|
overrides = DocumentMetadataOverrides()
|
|
|
|
# read all barcodes in the current document
|
|
if settings.CONSUMER_ENABLE_BARCODES or settings.CONSUMER_ENABLE_ASN_BARCODE:
|
|
doc_barcode_info = barcodes.scan_file_for_barcodes(
|
|
input_doc.original_file,
|
|
input_doc.mime_type,
|
|
)
|
|
|
|
# split document by separator pages, if enabled
|
|
if settings.CONSUMER_ENABLE_BARCODES:
|
|
separators = barcodes.get_separating_barcodes(doc_barcode_info.barcodes)
|
|
|
|
if len(separators) > 0:
|
|
logger.debug(
|
|
f"Pages with separators found in: {input_doc.original_file}",
|
|
)
|
|
document_list = barcodes.separate_pages(
|
|
doc_barcode_info.pdf_path,
|
|
separators,
|
|
)
|
|
|
|
if document_list:
|
|
|
|
# If the file is an upload, it's in the scratch directory
|
|
# Move it to consume directory to be picked up
|
|
# Otherwise, use the current parent to keep possible tags
|
|
# from subdirectories
|
|
if input_doc.source != DocumentSource.ConsumeFolder:
|
|
save_to_dir = settings.CONSUMPTION_DIR
|
|
else:
|
|
# Note this uses the original file, because it's in the
|
|
# consume folder already and may include additional path
|
|
# components for tagging
|
|
# the .path is somewhere in scratch in this case
|
|
save_to_dir = input_doc.original_file.parent
|
|
|
|
for n, document in enumerate(document_list):
|
|
# save to consumption dir
|
|
# rename it to the original filename with number prefix
|
|
if overrides.filename is not None:
|
|
newname = f"{str(n)}_{overrides.filename}"
|
|
else:
|
|
newname = None
|
|
|
|
barcodes.save_to_dir(
|
|
document,
|
|
newname=newname,
|
|
target_dir=save_to_dir,
|
|
)
|
|
|
|
# Split file has been copied safely, remove it
|
|
document.unlink()
|
|
|
|
# And clean up the directory as well, now it's empty
|
|
shutil.rmtree(document_list[0].parent)
|
|
|
|
# This file has been split into multiple files without issue
|
|
# remove the original and working copy
|
|
input_doc.original_file.unlink()
|
|
|
|
# If the original file was a TIFF, remove the PDF generated from it
|
|
if input_doc.mime_type == "image/tiff":
|
|
logger.debug(
|
|
f"Deleting file {doc_barcode_info.pdf_path}",
|
|
)
|
|
doc_barcode_info.pdf_path.unlink()
|
|
|
|
# notify the sender, otherwise the progress bar
|
|
# in the UI stays stuck
|
|
payload = {
|
|
"filename": overrides.filename or input_doc.original_file.name,
|
|
"task_id": None,
|
|
"current_progress": 100,
|
|
"max_progress": 100,
|
|
"status": "SUCCESS",
|
|
"message": "finished",
|
|
}
|
|
try:
|
|
async_to_sync(get_channel_layer().group_send)(
|
|
"status_updates",
|
|
{"type": "status_update", "data": payload},
|
|
)
|
|
except ConnectionError as e:
|
|
logger.warning(f"ConnectionError on status send: {str(e)}")
|
|
# consuming stops here, since the original document with
|
|
# the barcodes has been split and will be consumed separately
|
|
return "File successfully split"
|
|
|
|
# try reading the ASN from barcode
|
|
if settings.CONSUMER_ENABLE_ASN_BARCODE:
|
|
overrides.asn = barcodes.get_asn_from_barcodes(doc_barcode_info.barcodes)
|
|
if overrides.asn:
|
|
logger.info(f"Found ASN in barcode: {overrides.asn}")
|
|
|
|
# continue with consumption if no barcode was found
|
|
document = Consumer().try_consume_file(
|
|
input_doc.original_file,
|
|
override_filename=overrides.filename,
|
|
override_title=overrides.title,
|
|
override_correspondent_id=overrides.correspondent_id,
|
|
override_document_type_id=overrides.document_type_id,
|
|
override_tag_ids=overrides.tag_ids,
|
|
override_created=overrides.created,
|
|
override_asn=overrides.asn,
|
|
override_owner_id=overrides.owner_id,
|
|
)
|
|
|
|
if document:
|
|
return f"Success. New document id {document.pk} created"
|
|
else:
|
|
raise ConsumerError(
|
|
"Unknown error: Returned document was null, but "
|
|
"no error message was given.",
|
|
)
|
|
|
|
|
|
@shared_task
|
|
def sanity_check():
|
|
messages = sanity_checker.check_sanity()
|
|
|
|
messages.log_messages()
|
|
|
|
if messages.has_error:
|
|
raise SanityCheckFailedException("Sanity check failed with errors. See log.")
|
|
elif messages.has_warning:
|
|
return "Sanity check exited with warnings. See log."
|
|
elif len(messages) > 0:
|
|
return "Sanity check exited with infos. See log."
|
|
else:
|
|
return "No issues detected."
|
|
|
|
|
|
@shared_task
|
|
def bulk_update_documents(document_ids):
|
|
documents = Document.objects.filter(id__in=document_ids)
|
|
|
|
ix = index.open_index()
|
|
|
|
for doc in documents:
|
|
post_save.send(Document, instance=doc, created=False)
|
|
|
|
with AsyncWriter(ix) as writer:
|
|
for doc in documents:
|
|
index.update_document(writer, doc)
|
|
|
|
|
|
@shared_task
|
|
def update_document_archive_file(document_id):
|
|
"""
|
|
Re-creates the archive file of a document, including new OCR content and thumbnail
|
|
"""
|
|
document = Document.objects.get(id=document_id)
|
|
|
|
mime_type = document.mime_type
|
|
|
|
parser_class: Type[DocumentParser] = get_parser_class_for_mime_type(mime_type)
|
|
|
|
if not parser_class:
|
|
logger.error(
|
|
f"No parser found for mime type {mime_type}, cannot "
|
|
f"archive document {document} (ID: {document_id})",
|
|
)
|
|
return
|
|
|
|
parser: DocumentParser = parser_class(logging_group=uuid.uuid4())
|
|
|
|
try:
|
|
parser.parse(document.source_path, mime_type, document.get_public_filename())
|
|
|
|
thumbnail = parser.get_thumbnail(
|
|
document.source_path,
|
|
mime_type,
|
|
document.get_public_filename(),
|
|
)
|
|
|
|
if parser.get_archive_path():
|
|
with transaction.atomic():
|
|
with open(parser.get_archive_path(), "rb") as f:
|
|
checksum = hashlib.md5(f.read()).hexdigest()
|
|
# I'm going to save first so that in case the file move
|
|
# fails, the database is rolled back.
|
|
# We also don't use save() since that triggers the filehandling
|
|
# logic, and we don't want that yet (file not yet in place)
|
|
document.archive_filename = generate_unique_filename(
|
|
document,
|
|
archive_filename=True,
|
|
)
|
|
Document.objects.filter(pk=document.pk).update(
|
|
archive_checksum=checksum,
|
|
content=parser.get_text(),
|
|
archive_filename=document.archive_filename,
|
|
)
|
|
with FileLock(settings.MEDIA_LOCK):
|
|
create_source_path_directory(document.archive_path)
|
|
shutil.move(parser.get_archive_path(), document.archive_path)
|
|
shutil.move(thumbnail, document.thumbnail_path)
|
|
|
|
with index.open_index_writer() as writer:
|
|
index.update_document(writer, document)
|
|
|
|
except Exception:
|
|
logger.exception(
|
|
f"Error while parsing document {document} " f"(ID: {document_id})",
|
|
)
|
|
finally:
|
|
parser.cleanup()
|