mirror of
https://github.com/paperless-ngx/paperless-ngx.git
synced 2025-04-02 13:45:10 -05:00
Merge branch 'dev' into feature-websockets-status
This commit is contained in:
commit
46ea86a6d2
46
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
46
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
---
|
||||||
|
name: Bug report
|
||||||
|
about: Something is not working
|
||||||
|
title: "[BUG] Concise description of the issue"
|
||||||
|
labels: ''
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<!---
|
||||||
|
=> Before opening an issue, please check the documentation and see if it helps you resolve your issue: https://paperless-ng.readthedocs.io/en/latest/troubleshooting.html
|
||||||
|
=> Please also make sure that you followed the installation instructions.
|
||||||
|
=> Please search the issues and look for similar issues before opening a bug report.
|
||||||
|
|
||||||
|
=> If you encounter issues while installing of configuring Paperless-ng, please post that in the "Support" section of the discussions. Remember that Paperless successfully runs on a variety of different systems. If paperless does not start, it's probably an issue with your system, and not an issue of paperless.
|
||||||
|
|
||||||
|
=> Don't remove the [BUG] prefix from the title.
|
||||||
|
-->
|
||||||
|
|
||||||
|
**Describe the bug**
|
||||||
|
A clear and concise description of what the bug is.
|
||||||
|
|
||||||
|
**To Reproduce**
|
||||||
|
Steps to reproduce the behavior:
|
||||||
|
1. Go to '...'
|
||||||
|
2. Click on '....'
|
||||||
|
3. Scroll down to '....'
|
||||||
|
4. See error
|
||||||
|
|
||||||
|
**Expected behavior**
|
||||||
|
A clear and concise description of what you expected to happen.
|
||||||
|
|
||||||
|
**Screenshots**
|
||||||
|
If applicable, add screenshots to help explain your problem.
|
||||||
|
|
||||||
|
**Webserver logs**
|
||||||
|
```
|
||||||
|
If available, post any logs from the web server related to your issue.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Relevant information**
|
||||||
|
- Host OS of the machine running paperless: [e.g. Archlinux / Ubuntu 20.04]
|
||||||
|
- Browser [e.g. chrome, safari]
|
||||||
|
- Version [e.g. 1.0.0]
|
||||||
|
- Installation method: [docker / bare metal]
|
||||||
|
- Any configuration changes you made in `docker-compose.yml`, `docker-compose.env` or `paperless.conf`.
|
21
.github/ISSUE_TEMPLATE/feature_request.md
vendored
Normal file
21
.github/ISSUE_TEMPLATE/feature_request.md
vendored
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
---
|
||||||
|
name: Feature request
|
||||||
|
about: Suggest an idea for this project
|
||||||
|
title: "[Feature Request] Consice and clear description of your feature request"
|
||||||
|
labels: ''
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<!--
|
||||||
|
=> We already have lots of feature requests open. Please search the existing requests first and look for feature requests similar to yours.
|
||||||
|
|
||||||
|
=> Don't remove the [Feature Request] prefix from the title.
|
||||||
|
|
||||||
|
-->
|
||||||
|
|
||||||
|
**Is your feature request related to a problem? Please describe.**
|
||||||
|
A clear and concise description of what the problem is. Ex. I'm always frustrated when [...]
|
||||||
|
|
||||||
|
**Describe the solution you'd like**
|
||||||
|
A clear and concise description of what you want to happen.
|
18
.github/ISSUE_TEMPLATE/other.md
vendored
Normal file
18
.github/ISSUE_TEMPLATE/other.md
vendored
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
---
|
||||||
|
name: Other
|
||||||
|
about: Anything that is not a feature request or bug.
|
||||||
|
title: "[Other] Title of your issue"
|
||||||
|
labels: ''
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<!--
|
||||||
|
|
||||||
|
=> Discussions, Feedback and other suggestions belong in the "Disussion" section and not on the issue tracker.
|
||||||
|
|
||||||
|
=> If you encounter issues while installing of configuring Paperless-ng, please post that in the "Support" section of the discussions. Remember that Paperless successfully runs on a variety of different systems. If paperless does not start, it's probably is an issue with your system, and not an issue of paperless.
|
||||||
|
|
||||||
|
=> Don't remove the [Other] prefix from the title.
|
||||||
|
|
||||||
|
-->
|
16
.github/workflows/ansible.yml
vendored
16
.github/workflows/ansible.yml
vendored
@ -21,20 +21,20 @@ jobs:
|
|||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: |
|
run: |
|
||||||
python3 -m pip install --upgrade pip
|
python3 -m pip install --upgrade pip
|
||||||
python3 -m pip install molecule[ansible,docker]
|
python3 -m pip install molecule[ansible,docker] jmespath
|
||||||
ansible --version
|
ansible --version
|
||||||
docker --version
|
docker --version
|
||||||
molecule --version
|
molecule --version
|
||||||
python --version
|
python --version
|
||||||
- name: Test fresh installation with molecule
|
- name: Test installation/build/upgrade with molecule
|
||||||
run: |
|
run: |
|
||||||
cd ansible
|
cd ansible
|
||||||
molecule test -s fresh
|
molecule create
|
||||||
working-directory: "${{ github.repository }}"
|
molecule verify
|
||||||
- name: Test release update with molecule
|
molecule converge
|
||||||
run: |
|
molecule idempotence
|
||||||
cd ansible
|
molecule verify
|
||||||
molecule test -s update
|
molecule destroy
|
||||||
working-directory: "${{ github.repository }}"
|
working-directory: "${{ github.repository }}"
|
||||||
# # https://galaxy.ansible.com/docs/contributing/importing.html
|
# # https://galaxy.ansible.com/docs/contributing/importing.html
|
||||||
# release:
|
# release:
|
||||||
|
29
README.md
29
README.md
@ -1,4 +1,4 @@
|
|||||||

|
[](https://github.com/jonaswinkler/paperless-ng/actions)
|
||||||
[](https://paperless-ng.readthedocs.io/en/latest/?badge=latest)
|
[](https://paperless-ng.readthedocs.io/en/latest/?badge=latest)
|
||||||
[](https://gitter.im/paperless-ng/community?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge)
|
[](https://gitter.im/paperless-ng/community?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge)
|
||||||
[](https://hub.docker.com/r/jonaswinkler/paperless-ng)
|
[](https://hub.docker.com/r/jonaswinkler/paperless-ng)
|
||||||
@ -8,7 +8,11 @@
|
|||||||
|
|
||||||
[Paperless](https://github.com/the-paperless-project/paperless) is an application by Daniel Quinn and contributors that indexes your scanned documents and allows you to easily search for documents and store metadata alongside your documents.
|
[Paperless](https://github.com/the-paperless-project/paperless) is an application by Daniel Quinn and contributors that indexes your scanned documents and allows you to easily search for documents and store metadata alongside your documents.
|
||||||
|
|
||||||
Paperless-ng is a fork of the original project, adding a new interface and many other changes under the hood. For a detailed list of changes, have a look at the changelog in the documentation.
|
Paperless-ng is a fork of the original project, adding a new interface and many other changes under the hood. For a detailed list of changes, have a look at the [change log](https://paperless-ng.readthedocs.io/en/latest/changelog.html) in the documentation.
|
||||||
|
|
||||||
|
# Survey
|
||||||
|
|
||||||
|
If you already used Paperless-ng for a bit, would like to give some anonymous feedback, and help me decide on what to focus on next: I've created a survey, [see here](https://github.com/jonaswinkler/paperless-ng/issues/402). Thank you!
|
||||||
|
|
||||||
# How it Works
|
# How it Works
|
||||||
|
|
||||||
@ -29,6 +33,8 @@ Here's what you get:
|
|||||||
# Features
|
# Features
|
||||||
|
|
||||||
* Performs OCR on your documents, adds selectable text to image only documents and adds tags, correspondents and document types to your documents.
|
* Performs OCR on your documents, adds selectable text to image only documents and adds tags, correspondents and document types to your documents.
|
||||||
|
* Supports PDF documents, images, plain text files, and Office documents (Word, Excel, Powerpoint, and LibreOffice equivalents).
|
||||||
|
* Office document support is optional and provided by Apache Tika (see [configuration](https://paperless-ng.readthedocs.io/en/latest/configuration.html#tika-settings))
|
||||||
* Paperless stores your documents plain on disk. Filenames and folders are managed by paperless and can be configured freely.
|
* Paperless stores your documents plain on disk. Filenames and folders are managed by paperless and can be configured freely.
|
||||||
* Single page application front end. Should be pretty snappy. Will be mobile friendly in the future.
|
* Single page application front end. Should be pretty snappy. Will be mobile friendly in the future.
|
||||||
* Includes a dashboard that shows basic statistics and has document upload.
|
* Includes a dashboard that shows basic statistics and has document upload.
|
||||||
@ -50,25 +56,6 @@ If you want to see some screenshots of paperless-ng in action, [some are availab
|
|||||||
|
|
||||||
For a complete list of changes from paperless, check out the [changelog](https://paperless-ng.readthedocs.io/en/latest/changelog.html)
|
For a complete list of changes from paperless, check out the [changelog](https://paperless-ng.readthedocs.io/en/latest/changelog.html)
|
||||||
|
|
||||||
# Roadmap for 1.0
|
|
||||||
|
|
||||||
- Make the front end nice (except mobile).
|
|
||||||
- Fix whatever bugs I and you find.
|
|
||||||
- Make the documentation nice.
|
|
||||||
|
|
||||||
## On the chopping block.
|
|
||||||
|
|
||||||
- **GnuPG encrypion.** [Here's a note about encryption in paperless](https://paperless-ng.readthedocs.io/en/latest/administration.html#managing-encryption). The gist of it is that I don't see which attacks this implementation protects against. It gives a false sense of security to users who don't care about how it works.
|
|
||||||
|
|
||||||
## Wont-do list.
|
|
||||||
|
|
||||||
These features will probably never make it into paperless, since paperless is meant to be an easy to use set-and-forget solution.
|
|
||||||
|
|
||||||
- **Document versions.** I might consider adding the ability to update a document with a newer version, but that's about it. The kind of documents that get added to paperless usually don't change at all.
|
|
||||||
- **Workflows.** I don't see a use case for these, yet.
|
|
||||||
- **Folders.** Tags are superior in just about every way.
|
|
||||||
- **Apps / extension support.** Again, paperless is meant to be simple.
|
|
||||||
|
|
||||||
# Getting started
|
# Getting started
|
||||||
|
|
||||||
The recommended way to deploy paperless is docker-compose. The files in the /docker/hub directory are configured to pull the image from Docker Hub.
|
The recommended way to deploy paperless is docker-compose. The files in the /docker/hub directory are configured to pull the image from Docker Hub.
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
---
|
---
|
||||||
paperlessng_version: 0.9.14
|
paperlessng_version: latest # 'latest', release number, or github branch/tag/commit/ref
|
||||||
|
|
||||||
# Required services
|
# Required services
|
||||||
paperlessng_redis_host: localhost
|
paperlessng_redis_host: localhost
|
||||||
|
@ -2,10 +2,9 @@
|
|||||||
- name: update previous release to newest release
|
- name: update previous release to newest release
|
||||||
hosts: all
|
hosts: all
|
||||||
tasks:
|
tasks:
|
||||||
- name: set current version as installation target
|
- name: set github ref as version when available
|
||||||
set_fact:
|
set_fact:
|
||||||
paperlessng_version: 0.9.14
|
paperlessng_version: "{{ lookup('env', 'GITHUB_REF') | default('latest', True) }}"
|
||||||
|
|
||||||
- name: update to newest paperless-ng release
|
- name: update to newest paperless-ng release
|
||||||
include_role:
|
include_role:
|
||||||
name: ansible
|
name: ansible
|
@ -3,7 +3,7 @@
|
|||||||
tasks:
|
tasks:
|
||||||
- name: set previous version as installation target
|
- name: set previous version as installation target
|
||||||
set_fact:
|
set_fact:
|
||||||
paperlessng_version: 0.9.13
|
paperlessng_version: latest
|
||||||
|
|
||||||
- name: install previous paperless-ng release
|
- name: install previous paperless-ng release
|
||||||
include_role:
|
include_role:
|
94
ansible/molecule/default/verify.yml
Normal file
94
ansible/molecule/default/verify.yml
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
---
|
||||||
|
- name: Verify
|
||||||
|
hosts: all
|
||||||
|
gather_facts: false
|
||||||
|
|
||||||
|
vars_files:
|
||||||
|
- ../../defaults/main.yml
|
||||||
|
|
||||||
|
tasks:
|
||||||
|
- name: check if webserver is up
|
||||||
|
uri:
|
||||||
|
url: "http://{{ paperlessng_listen_address }}:{{ paperlessng_listen_port }}"
|
||||||
|
status_code: [200, 302]
|
||||||
|
return_content: yes
|
||||||
|
register: landingpage
|
||||||
|
failed_when: "'Sign in</button>' not in landingpage.content"
|
||||||
|
|
||||||
|
- name: generate random name and content
|
||||||
|
set_fact:
|
||||||
|
content: "{{ lookup('password', '/dev/null length=65536 chars=ascii_letters') }}"
|
||||||
|
filename: "{{ lookup('password', '/dev/null length=8 chars=ascii_letters') }}"
|
||||||
|
|
||||||
|
- name: check if document posting works
|
||||||
|
uri:
|
||||||
|
url: "http://{{ paperlessng_listen_address }}:{{ paperlessng_listen_port }}/api/documents/post_document/"
|
||||||
|
method: POST
|
||||||
|
body_format: form-multipart
|
||||||
|
body:
|
||||||
|
document:
|
||||||
|
content: "{{ content }}"
|
||||||
|
filename: "{{ filename }}.txt"
|
||||||
|
headers:
|
||||||
|
Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}'
|
||||||
|
Content-Type: text/plain
|
||||||
|
return_content: yes
|
||||||
|
register: post_document
|
||||||
|
failed_when: "'OK' not in post_document.content"
|
||||||
|
|
||||||
|
- name: verify uploaded document has been accepted
|
||||||
|
uri:
|
||||||
|
url: "http://{{ paperlessng_listen_address }}:{{ paperlessng_listen_port }}/api/logs/"
|
||||||
|
headers:
|
||||||
|
Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}'
|
||||||
|
return_content: yes
|
||||||
|
register: logs
|
||||||
|
failed_when: "('Consuming ' + filename + '.txt') not in logs.content"
|
||||||
|
|
||||||
|
- name: sleep till consumption finished
|
||||||
|
pause:
|
||||||
|
seconds: 10
|
||||||
|
|
||||||
|
- name: verify uploaded document has been consumed
|
||||||
|
uri:
|
||||||
|
url: "http://{{ paperlessng_listen_address }}:{{ paperlessng_listen_port }}/api/logs/"
|
||||||
|
headers:
|
||||||
|
Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}'
|
||||||
|
return_content: yes
|
||||||
|
register: logs
|
||||||
|
failed_when: "filename + ' consumption finished' not in logs.content"
|
||||||
|
|
||||||
|
- name: get documents
|
||||||
|
uri:
|
||||||
|
url: "http://{{ paperlessng_listen_address }}:{{ paperlessng_listen_port }}/api/documents/"
|
||||||
|
headers:
|
||||||
|
Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}'
|
||||||
|
return_content: yes
|
||||||
|
register: documents
|
||||||
|
|
||||||
|
- name: set document index
|
||||||
|
set_fact:
|
||||||
|
index: "{{ documents.json['results'][0]['id'] }}"
|
||||||
|
|
||||||
|
- name: verify uploaded document is avaiable
|
||||||
|
uri:
|
||||||
|
url: "http://{{ paperlessng_listen_address }}:{{ paperlessng_listen_port }}/api/documents/{{ index }}/"
|
||||||
|
headers:
|
||||||
|
Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}'
|
||||||
|
return_content: yes
|
||||||
|
register: document
|
||||||
|
failed_when: "'Not found.' in document.content or content not in document.json['content']"
|
||||||
|
|
||||||
|
- name: check if deleting uploaded document works
|
||||||
|
uri:
|
||||||
|
url: "http://{{ paperlessng_listen_address }}:{{ paperlessng_listen_port }}/api/documents/bulk_edit/"
|
||||||
|
method: POST
|
||||||
|
body_format: json
|
||||||
|
body:
|
||||||
|
documents: ["{{ index }}"]
|
||||||
|
method: delete
|
||||||
|
parameters: {}
|
||||||
|
headers:
|
||||||
|
Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}'
|
||||||
|
register: delete_document
|
||||||
|
failed_when: "'OK' not in delete_document.json['result']"
|
@ -1,7 +0,0 @@
|
|||||||
---
|
|
||||||
- name: fresh installation
|
|
||||||
hosts: all
|
|
||||||
tasks:
|
|
||||||
- name: install paperless-ng with default parameters
|
|
||||||
include_role:
|
|
||||||
name: ansible
|
|
@ -1,60 +0,0 @@
|
|||||||
---
|
|
||||||
- name: Verify
|
|
||||||
hosts: all
|
|
||||||
gather_facts: false
|
|
||||||
|
|
||||||
vars_files:
|
|
||||||
- ../../defaults/main.yml
|
|
||||||
|
|
||||||
tasks:
|
|
||||||
- name: check if webserver is up
|
|
||||||
uri:
|
|
||||||
url: http://localhost:8000
|
|
||||||
status_code: [200, 302]
|
|
||||||
return_content: yes
|
|
||||||
register: landingpage
|
|
||||||
failed_when: "'Sign in</button>' not in landingpage.content"
|
|
||||||
|
|
||||||
- name: check if document posting works
|
|
||||||
uri:
|
|
||||||
url: http://localhost:8000/api/documents/post_document/
|
|
||||||
method: POST
|
|
||||||
body_format: form-multipart
|
|
||||||
body:
|
|
||||||
document:
|
|
||||||
content: FOO
|
|
||||||
filename: document.txt
|
|
||||||
mime_type: text/plain
|
|
||||||
headers:
|
|
||||||
Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}'
|
|
||||||
return_content: yes
|
|
||||||
register: post_document
|
|
||||||
failed_when: "'OK' not in post_document.content"
|
|
||||||
|
|
||||||
- name: verify uploaded document has been accepted
|
|
||||||
uri:
|
|
||||||
url: http://localhost:8000/api/logs/
|
|
||||||
headers:
|
|
||||||
Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}'
|
|
||||||
return_content: yes
|
|
||||||
register: logs
|
|
||||||
failed_when: "'Consuming document.txt' not in logs.content"
|
|
||||||
|
|
||||||
# assumes txt consumption finished by now, might have to sleep a bit
|
|
||||||
- name: verify uploaded document has been consumed
|
|
||||||
uri:
|
|
||||||
url: http://localhost:8000/api/logs/
|
|
||||||
headers:
|
|
||||||
Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}'
|
|
||||||
return_content: yes
|
|
||||||
register: logs
|
|
||||||
failed_when: "'document consumption finished' not in logs.content"
|
|
||||||
|
|
||||||
- name: verify uploaded document is avaiable
|
|
||||||
uri:
|
|
||||||
url: http://localhost:8000/api/documents/1/
|
|
||||||
headers:
|
|
||||||
Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}'
|
|
||||||
return_content: yes
|
|
||||||
register: document
|
|
||||||
failed_when: "'Not found.' in document.content or 'FOO' not in document.content"
|
|
@ -1,35 +0,0 @@
|
|||||||
---
|
|
||||||
dependency:
|
|
||||||
name: galaxy
|
|
||||||
driver:
|
|
||||||
name: docker
|
|
||||||
platforms:
|
|
||||||
- name: ubuntu_focal
|
|
||||||
image: jrei/systemd-ubuntu:20.04
|
|
||||||
privileged: true
|
|
||||||
volumes:
|
|
||||||
- /sys/fs/cgroup:/sys/fs/cgroup:ro
|
|
||||||
tmpfs:
|
|
||||||
- /tmp
|
|
||||||
- /run
|
|
||||||
- /run/lock
|
|
||||||
override_command: False
|
|
||||||
# ubuntu 18.04 bionic works except that
|
|
||||||
# the default redis configuration expects IPv6 which is not enabled in docker by default
|
|
||||||
# the default Python environment is configured for ASCII instead of UTF-8
|
|
||||||
# ubuntu 16.04 xenial only has Python 3.5 which is EOL and breaks multiple dependencies
|
|
||||||
- name: debian_buster
|
|
||||||
image: jrei/systemd-debian:10
|
|
||||||
privileged: true
|
|
||||||
volumes:
|
|
||||||
- /sys/fs/cgroup:/sys/fs/cgroup:ro
|
|
||||||
tmpfs:
|
|
||||||
- /tmp
|
|
||||||
- /run
|
|
||||||
- /run/lock
|
|
||||||
override_command: False
|
|
||||||
# debian 9 stretch only has Python 3.5 which is EOL and breaks multiple dependencies
|
|
||||||
provisioner:
|
|
||||||
name: ansible
|
|
||||||
verifier:
|
|
||||||
name: ansible
|
|
@ -1,60 +0,0 @@
|
|||||||
---
|
|
||||||
- name: Verify
|
|
||||||
hosts: all
|
|
||||||
gather_facts: false
|
|
||||||
|
|
||||||
vars_files:
|
|
||||||
- ../../defaults/main.yml
|
|
||||||
|
|
||||||
tasks:
|
|
||||||
- name: check if webserver is up
|
|
||||||
uri:
|
|
||||||
url: http://localhost:8000
|
|
||||||
status_code: [200, 302]
|
|
||||||
return_content: yes
|
|
||||||
register: landingpage
|
|
||||||
failed_when: "'Sign in</button>' not in landingpage.content"
|
|
||||||
|
|
||||||
- name: check if document posting works
|
|
||||||
uri:
|
|
||||||
url: http://localhost:8000/api/documents/post_document/
|
|
||||||
method: POST
|
|
||||||
body_format: form-multipart
|
|
||||||
body:
|
|
||||||
document:
|
|
||||||
content: FOO
|
|
||||||
filename: document.txt
|
|
||||||
mime_type: text/plain
|
|
||||||
headers:
|
|
||||||
Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}'
|
|
||||||
return_content: yes
|
|
||||||
register: post_document
|
|
||||||
failed_when: "'OK' not in post_document.content"
|
|
||||||
|
|
||||||
- name: verify uploaded document has been accepted
|
|
||||||
uri:
|
|
||||||
url: http://localhost:8000/api/logs/
|
|
||||||
headers:
|
|
||||||
Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}'
|
|
||||||
return_content: yes
|
|
||||||
register: logs
|
|
||||||
failed_when: "'Consuming document.txt' not in logs.content"
|
|
||||||
|
|
||||||
# assumes txt consumption finished by now, might have to sleep a bit
|
|
||||||
- name: verify uploaded document has been consumed
|
|
||||||
uri:
|
|
||||||
url: http://localhost:8000/api/logs/
|
|
||||||
headers:
|
|
||||||
Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}'
|
|
||||||
return_content: yes
|
|
||||||
register: logs
|
|
||||||
failed_when: "'document consumption finished' not in logs.content"
|
|
||||||
|
|
||||||
- name: verify uploaded document is avaiable
|
|
||||||
uri:
|
|
||||||
url: http://localhost:8000/api/documents/1/
|
|
||||||
headers:
|
|
||||||
Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}'
|
|
||||||
return_content: yes
|
|
||||||
register: document
|
|
||||||
failed_when: "'Not found.' in document.content or 'FOO' not in document.content"
|
|
6
ansible/tasks/install-release.yml
Normal file
6
ansible/tasks/install-release.yml
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
---
|
||||||
|
- name: extract paperless-ng
|
||||||
|
unarchive:
|
||||||
|
src: "https://github.com/jonaswinkler/paperless-ng/releases/download/ng-{{ paperlessng_version }}/paperless-ng-{{ paperlessng_version }}.tar.xz"
|
||||||
|
remote_src: yes
|
||||||
|
dest: "{{ tempdir.path }}"
|
111
ansible/tasks/install-source.yml
Normal file
111
ansible/tasks/install-source.yml
Normal file
@ -0,0 +1,111 @@
|
|||||||
|
---
|
||||||
|
- name: install dev dependencies
|
||||||
|
apt:
|
||||||
|
pkg:
|
||||||
|
- git
|
||||||
|
- npm
|
||||||
|
- gettext
|
||||||
|
|
||||||
|
- name: create output directories
|
||||||
|
file:
|
||||||
|
path: "{{ item }}"
|
||||||
|
state: directory
|
||||||
|
owner: "{{ paperlessng_system_user }}"
|
||||||
|
group: "{{ paperlessng_system_group }}"
|
||||||
|
mode: "750"
|
||||||
|
with_items:
|
||||||
|
- "{{ tempdir.path }}/paperless-ng"
|
||||||
|
- "{{ tempdir.path }}/paperless-ng/scripts"
|
||||||
|
|
||||||
|
- block:
|
||||||
|
- name: create temporary git directory
|
||||||
|
tempfile:
|
||||||
|
state: directory
|
||||||
|
path: "{{ paperlessng_directory }}"
|
||||||
|
register: gitdir
|
||||||
|
|
||||||
|
- name: pull paperless-ng
|
||||||
|
git:
|
||||||
|
repo: https://github.com/jonaswinkler/paperless-ng.git
|
||||||
|
dest: "{{ gitdir.path }}"
|
||||||
|
version: "{{ paperlessng_version }}"
|
||||||
|
refspec: "+refs/pull/*:refs/pull/*"
|
||||||
|
|
||||||
|
- name: compile frontend
|
||||||
|
command:
|
||||||
|
cmd: "{{ item }}"
|
||||||
|
args:
|
||||||
|
chdir: "{{ gitdir.path }}/src-ui"
|
||||||
|
failed_when: false
|
||||||
|
with_items:
|
||||||
|
- npm install -g @angular/cli
|
||||||
|
- npm install
|
||||||
|
- ./node_modules/.bin/ng build --prod
|
||||||
|
|
||||||
|
- name: copy application into place
|
||||||
|
copy:
|
||||||
|
src: "{{ gitdir.path }}/{{ item.src }}"
|
||||||
|
remote_src: yes
|
||||||
|
dest: "{{ tempdir.path }}/paperless-ng/{{ item.dest | default('') }}"
|
||||||
|
with_items:
|
||||||
|
- src: CONTRIBUTING.md
|
||||||
|
- src: LICENSE
|
||||||
|
- src: Pipfile
|
||||||
|
- src: Pipfile.lock
|
||||||
|
- src: README.md
|
||||||
|
- src: requirements.txt
|
||||||
|
- src: paperless.conf.example
|
||||||
|
dest: "paperless.conf"
|
||||||
|
|
||||||
|
- name: glob all scripts
|
||||||
|
find:
|
||||||
|
paths: ["{{ gitdir.path }}/scripts/"]
|
||||||
|
patterns:
|
||||||
|
- "*.service"
|
||||||
|
- "*.sh"
|
||||||
|
register: glob
|
||||||
|
|
||||||
|
- name: copy scripts
|
||||||
|
copy:
|
||||||
|
src: "{{ item.path }}"
|
||||||
|
remote_src: yes
|
||||||
|
dest: "{{ tempdir.path }}/paperless-ng/scripts/"
|
||||||
|
with_items:
|
||||||
|
- "{{ glob.files }}"
|
||||||
|
|
||||||
|
- name: copy sources
|
||||||
|
command:
|
||||||
|
cmd: "cp -r src/ {{ tempdir.path }}/paperless-ng/src"
|
||||||
|
args:
|
||||||
|
chdir: "{{ gitdir.path }}"
|
||||||
|
|
||||||
|
- name: create paperlessng venv
|
||||||
|
command:
|
||||||
|
cmd: "python3 -m virtualenv {{ gitdir.path }}/.venv/ -p /usr/bin/python3"
|
||||||
|
|
||||||
|
- name: install paperlessng requirements
|
||||||
|
command:
|
||||||
|
cmd: "{{ gitdir.path }}/.venv/bin/python3 -m pip install -r {{ gitdir.path }}/requirements.txt"
|
||||||
|
|
||||||
|
- name: compile messages
|
||||||
|
command: "{{ gitdir.path }}/.venv/bin/python3 manage.py compilemessages"
|
||||||
|
args:
|
||||||
|
chdir: "{{ tempdir.path }}/paperless-ng/src/"
|
||||||
|
|
||||||
|
- name: collect static files
|
||||||
|
command: "{{ gitdir.path }}/.venv/bin/python3 manage.py collectstatic --no-input"
|
||||||
|
args:
|
||||||
|
chdir: "{{ tempdir.path }}/paperless-ng/src/"
|
||||||
|
|
||||||
|
- name: remove pycache directories
|
||||||
|
shell: find . -name __pycache__ | xargs rm -r
|
||||||
|
args:
|
||||||
|
chdir: "{{ tempdir.path }}"
|
||||||
|
|
||||||
|
- name: remove temporary git directory
|
||||||
|
file:
|
||||||
|
path: "{{ gitdir.path }}"
|
||||||
|
state: absent
|
||||||
|
|
||||||
|
become: yes
|
||||||
|
become_user: "{{ paperlessng_system_user }}"
|
@ -34,7 +34,13 @@
|
|||||||
- build-essential
|
- build-essential
|
||||||
- python3-setuptools
|
- python3-setuptools
|
||||||
- python3-wheel
|
- python3-wheel
|
||||||
- python3-virtualenv
|
|
||||||
|
# upstream virtualenv in Ubuntu 20.04 is broken
|
||||||
|
# https://github.com/pypa/virtualenv/issues/1873
|
||||||
|
- name: install python virtualenv
|
||||||
|
pip:
|
||||||
|
name: virtualenv
|
||||||
|
extra_args: --upgrade
|
||||||
|
|
||||||
- name: install ocr languages
|
- name: install ocr languages
|
||||||
apt:
|
apt:
|
||||||
@ -97,71 +103,141 @@
|
|||||||
# GNUPG_HOME required due to paperless db.py
|
# GNUPG_HOME required due to paperless db.py
|
||||||
create_home: yes
|
create_home: yes
|
||||||
|
|
||||||
|
- block:
|
||||||
|
- name: get latest release version
|
||||||
|
uri:
|
||||||
|
url: https://api.github.com/repos/jonaswinkler/paperless-ng/releases/latest
|
||||||
|
method: GET
|
||||||
|
register: latest_release
|
||||||
|
- name: parse latest release version
|
||||||
|
set_fact:
|
||||||
|
paperlessng_version: "{{ latest_release.json['tag_name'] }}"
|
||||||
|
when: paperlessng_version == "latest"
|
||||||
|
|
||||||
|
- block:
|
||||||
|
- name: sanitize version string
|
||||||
|
set_fact:
|
||||||
|
paperlessng_version: "{{ paperlessng_version | regex_replace('^ng-(\\d+\\.\\d+\\.\\d+)$', '\\1') }}"
|
||||||
|
- name: get tag data
|
||||||
|
uri:
|
||||||
|
url: https://api.github.com/repos/jonaswinkler/paperless-ng/tags
|
||||||
|
method: GET
|
||||||
|
register: tags
|
||||||
|
- name: get commit for target tag
|
||||||
|
set_fact:
|
||||||
|
paperlessng_commit: "{{ tags.json | json_query('[?name==`ng-' + paperlessng_version +'`] | [0].commit.sha') }}"
|
||||||
|
when: paperlessng_version | regex_search("^(ng-)?(\d+\.\d+\.\d+)$")
|
||||||
|
|
||||||
|
- block:
|
||||||
|
- name: check if version is branch
|
||||||
|
uri:
|
||||||
|
url: "https://api.github.com/repos/jonaswinkler/paperless-ng/branches/{{ paperlessng_version }}"
|
||||||
|
method: GET
|
||||||
|
status_code: [200, 404]
|
||||||
|
register: branch
|
||||||
|
- name: get commit for target branch
|
||||||
|
set_fact:
|
||||||
|
paperlessng_commit: "{{ branch.json | json_query('commit.sha') }}"
|
||||||
|
when: branch.status == 200
|
||||||
|
- block:
|
||||||
|
- name: check if version is commit-or-ref
|
||||||
|
uri:
|
||||||
|
url: "https://api.github.com/repos/jonaswinkler/paperless-ng/commits/{{ paperlessng_version }}"
|
||||||
|
method: GET
|
||||||
|
status_code: [200, 404, 422]
|
||||||
|
register: commit
|
||||||
|
- name: get commit for target commit-or-ref
|
||||||
|
set_fact:
|
||||||
|
paperlessng_commit: "{{ commit.json | json_query('sha') }}"
|
||||||
|
when: commit.status == 200
|
||||||
|
- name: fail
|
||||||
|
fail:
|
||||||
|
msg: "Can not determine commit from `paperlessng_version=={{ paperlessng_version }}`!"
|
||||||
|
when: commit.status != 200
|
||||||
|
when: branch.status == 404
|
||||||
|
when: not(paperlessng_version | regex_search("^(ng-)?(\d+\.\d+\.\d+)$"))
|
||||||
|
|
||||||
- name: check for paperless-ng installation
|
- name: check for paperless-ng installation
|
||||||
command:
|
command:
|
||||||
cmd: 'grep -Po "(?<=Paperless-ng )\d+\.\d+\.\d+" {{ paperlessng_directory }}/docs/changelog.html'
|
cmd: "cat {{ paperlessng_directory }}/.installed_version"
|
||||||
changed_when: '"No such file or directory" in paperlessng_current_version.stderr or paperlessng_current_version.stdout != paperlessng_version | string'
|
changed_when: '"No such file or directory" in paperlessng_current_commit.stderr or paperlessng_current_commit.stdout != paperlessng_commit | string'
|
||||||
failed_when: false
|
failed_when: false
|
||||||
ignore_errors: yes
|
ignore_errors: yes
|
||||||
register: paperlessng_current_version
|
register: paperlessng_current_commit
|
||||||
|
|
||||||
- name: register current state
|
- name: register current state
|
||||||
set_fact:
|
set_fact:
|
||||||
fresh_installation: '{{ "No such file or directory" in paperlessng_current_version.stderr }}'
|
fresh_installation: '{{ "No such file or directory" in paperlessng_current_commit.stderr }}'
|
||||||
update_installation: '{{ "No such file or directory" not in paperlessng_current_version.stderr and paperlessng_current_version.stdout != paperlessng_version | string }}'
|
update_installation: '{{ "No such file or directory" not in paperlessng_current_commit.stderr and paperlessng_current_commit.stdout != paperlessng_commit | string }}'
|
||||||
reconfigure_only: '{{ paperlessng_current_version.stdout == paperlessng_version | string }}'
|
reconfigure_only: "{{ paperlessng_current_commit.stdout == paperlessng_commit | string }}"
|
||||||
|
|
||||||
- name: backup current paperless-ng installation
|
- block:
|
||||||
copy:
|
- name: backup current paperless-ng installation
|
||||||
src: "{{ paperlessng_directory }}"
|
copy:
|
||||||
remote_src: yes
|
src: "{{ paperlessng_directory }}"
|
||||||
dest: "{{ paperlessng_directory }}-{{ ansible_date_time.iso8601 }}/"
|
remote_src: yes
|
||||||
|
dest: "{{ paperlessng_directory }}-{{ ansible_date_time.iso8601 }}/"
|
||||||
|
- name: remove current paperless sources
|
||||||
|
file:
|
||||||
|
path: "{{ paperlessng_directory }}/{{ item }}"
|
||||||
|
state: absent
|
||||||
|
with_items:
|
||||||
|
- docker
|
||||||
|
- docs
|
||||||
|
- scripts
|
||||||
|
- src
|
||||||
|
- static
|
||||||
when: update_installation
|
when: update_installation
|
||||||
|
|
||||||
- name: remove current paperless sources
|
- block:
|
||||||
file:
|
- name: create paperless-ng directory and set permissions
|
||||||
path: "{{ paperlessng_directory }}/{{ item }}"
|
file:
|
||||||
state: absent
|
path: "{{ paperlessng_directory }}"
|
||||||
with_items:
|
state: directory
|
||||||
- docker
|
owner: "{{ paperlessng_system_user }}"
|
||||||
- docs
|
group: "{{ paperlessng_system_group }}"
|
||||||
- scripts
|
mode: "750"
|
||||||
- src
|
- name: create temporary directory
|
||||||
- static
|
become: yes
|
||||||
when: update_installation
|
become_user: "{{ paperlessng_system_user }}"
|
||||||
|
tempfile:
|
||||||
- name: create temporary directory
|
state: directory
|
||||||
tempfile:
|
path: "{{ paperlessng_directory }}"
|
||||||
state: directory
|
register: tempdir
|
||||||
register: tempdir
|
- name: check if version is available as release archive
|
||||||
when: not reconfigure_only
|
uri:
|
||||||
|
url: "https://github.com/jonaswinkler/paperless-ng/releases/download/ng-{{ paperlessng_version }}/paperless-ng-{{ paperlessng_version }}.tar.xz"
|
||||||
- name: extract paperless-ng
|
method: GET
|
||||||
unarchive:
|
status_code: [200, 404]
|
||||||
src: "https://github.com/jonaswinkler/paperless-ng/releases/download/ng-{{ paperlessng_version }}/paperless-ng-{{ paperlessng_version }}.tar.xz"
|
register: release_archive
|
||||||
remote_src: yes
|
- name: install paperless-ng from source
|
||||||
dest: "{{ tempdir.path }}"
|
include_tasks: install-source.yml
|
||||||
when: not reconfigure_only
|
when: release_archive.status == 404
|
||||||
|
- name: install paperless-ng from release archive
|
||||||
- name: change owner and permissions of paperless-ng
|
include_tasks: install-release.yml
|
||||||
command:
|
when: release_archive.status == 200
|
||||||
cmd: "{{ item }}"
|
- name: change owner and permissions of paperless-ng
|
||||||
warn: false
|
command:
|
||||||
with_items:
|
cmd: "{{ item }}"
|
||||||
- "chown -R {{ paperlessng_system_user }}:{{ paperlessng_system_group }} {{ tempdir.path }}"
|
warn: false
|
||||||
- "find {{ tempdir.path }} -type d -exec chmod 0750 {} ;"
|
with_items:
|
||||||
- "find {{ tempdir.path }} -type f -exec chmod 0640 {} ;"
|
- "chown -R {{ paperlessng_system_user }}:{{ paperlessng_system_group }} {{ tempdir.path }}"
|
||||||
when: not reconfigure_only
|
- "find {{ tempdir.path }} -type d -exec chmod 0750 {} ;"
|
||||||
|
- "find {{ tempdir.path }} -type f -exec chmod 0640 {} ;"
|
||||||
- name: move paperless-ng
|
- name: move paperless-ng
|
||||||
command:
|
command:
|
||||||
cmd: "cp -a {{ tempdir.path }}/paperless-ng/. {{ paperlessng_directory }}"
|
cmd: "cp -a {{ tempdir.path }}/paperless-ng/. {{ paperlessng_directory }}"
|
||||||
when: not reconfigure_only
|
- name: store commit hash of installed version
|
||||||
|
copy:
|
||||||
- name: remove temporary directory
|
content: "{{ paperlessng_commit }}"
|
||||||
file:
|
dest: "{{ paperlessng_directory }}/.installed_version"
|
||||||
path: "{{ tempdir.path }}"
|
owner: "{{ paperlessng_system_user }}"
|
||||||
state: absent
|
group: "{{ paperlessng_system_group }}"
|
||||||
|
mode: "0440"
|
||||||
|
- name: remove temporary directory
|
||||||
|
file:
|
||||||
|
path: "{{ tempdir.path }}"
|
||||||
|
state: absent
|
||||||
when: not reconfigure_only
|
when: not reconfigure_only
|
||||||
|
|
||||||
- name: create paperless-ng directories and set permissions
|
- name: create paperless-ng directories and set permissions
|
||||||
@ -172,7 +248,6 @@
|
|||||||
group: "{{ paperlessng_system_group }}"
|
group: "{{ paperlessng_system_group }}"
|
||||||
mode: "750"
|
mode: "750"
|
||||||
with_items:
|
with_items:
|
||||||
- "{{ paperlessng_directory }}"
|
|
||||||
- "{{ paperlessng_consumption_dir }}"
|
- "{{ paperlessng_consumption_dir }}"
|
||||||
- "{{ paperlessng_data_dir }}"
|
- "{{ paperlessng_data_dir }}"
|
||||||
- "{{ paperlessng_media_root }}"
|
- "{{ paperlessng_media_root }}"
|
||||||
@ -180,7 +255,7 @@
|
|||||||
|
|
||||||
- name: rename initial config
|
- name: rename initial config
|
||||||
command:
|
command:
|
||||||
cmd: "mv {{ paperlessng_directory }}/paperless.conf {{ paperlessng_directory }}/paperless.conf.template"
|
cmd: "mv -f {{ paperlessng_directory }}/paperless.conf {{ paperlessng_directory }}/paperless.conf.template"
|
||||||
removes: "{{ paperlessng_directory }}/paperless.conf"
|
removes: "{{ paperlessng_directory }}/paperless.conf"
|
||||||
|
|
||||||
- name: configure paperless-ng
|
- name: configure paperless-ng
|
||||||
@ -310,21 +385,20 @@
|
|||||||
creates: "{{ paperlessng_virtualenv }}"
|
creates: "{{ paperlessng_virtualenv }}"
|
||||||
register: venv
|
register: venv
|
||||||
|
|
||||||
- name: install paperlessng requirements
|
- block:
|
||||||
become: yes
|
- name: install paperlessng requirements
|
||||||
become_user: "{{ paperlessng_system_user }}"
|
become: yes
|
||||||
pip:
|
become_user: "{{ paperlessng_system_user }}"
|
||||||
requirements: "{{ paperlessng_directory }}/requirements.txt"
|
pip:
|
||||||
executable: "{{ paperlessng_virtualenv }}/bin/pip3"
|
requirements: "{{ paperlessng_directory }}/requirements.txt"
|
||||||
extra_args: --upgrade
|
executable: "{{ paperlessng_virtualenv }}/bin/pip3"
|
||||||
when: not reconfigure_only
|
extra_args: --upgrade
|
||||||
|
- name: migrate database schema
|
||||||
- name: migrate database schema
|
become: yes
|
||||||
become: yes
|
become_user: "{{ paperlessng_system_user }}"
|
||||||
become_user: "{{ paperlessng_system_user }}"
|
command: "{{ paperlessng_virtualenv }}/bin/python3 {{ paperlessng_directory }}/src/manage.py migrate"
|
||||||
command: "{{ paperlessng_virtualenv }}/bin/python3 {{ paperlessng_directory }}/src/manage.py migrate"
|
register: database_schema
|
||||||
register: database_schema
|
changed_when: '"No migrations to apply." not in database_schema.stdout'
|
||||||
changed_when: '"No migrations to apply." not in database_schema.stdout'
|
|
||||||
when: not reconfigure_only
|
when: not reconfigure_only
|
||||||
|
|
||||||
- name: configure paperless superuser
|
- name: configure paperless superuser
|
||||||
@ -392,7 +466,7 @@
|
|||||||
# https://www.freedesktop.org/software/systemd/man/systemd.exec.html
|
# https://www.freedesktop.org/software/systemd/man/systemd.exec.html
|
||||||
{ option: "User", value: "{{ paperlessng_system_user }}" },
|
{ option: "User", value: "{{ paperlessng_system_user }}" },
|
||||||
{ option: "Group", value: "{{ paperlessng_system_group }}" },
|
{ option: "Group", value: "{{ paperlessng_system_group }}" },
|
||||||
{ option: "WorkingDirectory", value: "{{ paperlessng_directory }}/src", },
|
{ option: "WorkingDirectory", value: "{{ paperlessng_directory }}/src" },
|
||||||
{ option: "ProtectSystem", value: "full" },
|
{ option: "ProtectSystem", value: "full" },
|
||||||
{ option: "NoNewPrivileges", value: "true" },
|
{ option: "NoNewPrivileges", value: "true" },
|
||||||
{ option: "PrivateUsers", value: "true" },
|
{ option: "PrivateUsers", value: "true" },
|
||||||
|
@ -121,27 +121,19 @@ After grabbing the new release and unpacking the contents, do the following:
|
|||||||
dependencies. The dependencies required are listed in the section about
|
dependencies. The dependencies required are listed in the section about
|
||||||
:ref:`bare metal installations <setup-bare_metal>`.
|
:ref:`bare metal installations <setup-bare_metal>`.
|
||||||
|
|
||||||
2. Update python requirements. If you use Pipenv, this is done with the following steps.
|
2. Update python requirements. Keep in mind to activate your virtual environment
|
||||||
|
before that, if you use one.
|
||||||
|
|
||||||
.. code:: shell-session
|
.. code:: shell-session
|
||||||
|
|
||||||
$ pip install --upgrade pipenv
|
$ pip install -r requirements.txt
|
||||||
$ cd /path/to/paperless
|
|
||||||
$ pipenv clean
|
|
||||||
$ pipenv install
|
|
||||||
|
|
||||||
This creates a new virtual environment (or uses your existing environment)
|
|
||||||
and installs all dependencies into it.
|
|
||||||
|
|
||||||
You can also use the included ``requirements.txt`` file instead and create the virtual
|
|
||||||
environment yourself. This file includes exactly the same dependencies.
|
|
||||||
|
|
||||||
3. Migrate the database.
|
3. Migrate the database.
|
||||||
|
|
||||||
.. code:: shell-session
|
.. code:: shell-session
|
||||||
|
|
||||||
$ cd src
|
$ cd src
|
||||||
$ pipenv run python3 manage.py migrate
|
$ python3 manage.py migrate
|
||||||
|
|
||||||
This might not actually do anything. Not every new paperless version comes with new
|
This might not actually do anything. Not every new paperless version comes with new
|
||||||
database migrations.
|
database migrations.
|
||||||
@ -195,7 +187,7 @@ or
|
|||||||
.. code:: shell-session
|
.. code:: shell-session
|
||||||
|
|
||||||
$ cd /path/to/paperless/src
|
$ cd /path/to/paperless/src
|
||||||
$ pipenv run python manage.py <command> <arguments>
|
$ python3 manage.py <command> <arguments>
|
||||||
|
|
||||||
depending on whether you use docker or not.
|
depending on whether you use docker or not.
|
||||||
|
|
||||||
@ -462,6 +454,3 @@ Basic usage to disable encryption of your document store:
|
|||||||
.. code::
|
.. code::
|
||||||
|
|
||||||
decrypt_documents [--passphrase SECR3TP4SSPHRA$E]
|
decrypt_documents [--passphrase SECR3TP4SSPHRA$E]
|
||||||
|
|
||||||
|
|
||||||
.. _Pipenv: https://pipenv.pypa.io/en/latest/
|
|
||||||
|
@ -376,25 +376,24 @@ PAPERLESS_THREADS_PER_WORKER=<num>
|
|||||||
use a higher thread per worker count.
|
use a higher thread per worker count.
|
||||||
|
|
||||||
The default is a balance between the two, according to your CPU core count,
|
The default is a balance between the two, according to your CPU core count,
|
||||||
with a slight favor towards threads per worker, and leaving at least one core
|
with a slight favor towards threads per worker:
|
||||||
free for other tasks:
|
|
||||||
|
|
||||||
+----------------+---------+---------+
|
+----------------+---------+---------+
|
||||||
| CPU core count | Workers | Threads |
|
| CPU core count | Workers | Threads |
|
||||||
+----------------+---------+---------+
|
+----------------+---------+---------+
|
||||||
| 1 | 1 | 1 |
|
| 1 | 1 | 1 |
|
||||||
+----------------+---------+---------+
|
+----------------+---------+---------+
|
||||||
| 2 | 1 | 1 |
|
| 2 | 2 | 1 |
|
||||||
+----------------+---------+---------+
|
+----------------+---------+---------+
|
||||||
| 4 | 1 | 3 |
|
| 4 | 2 | 2 |
|
||||||
+----------------+---------+---------+
|
+----------------+---------+---------+
|
||||||
| 6 | 2 | 2 |
|
| 6 | 2 | 3 |
|
||||||
+----------------+---------+---------+
|
+----------------+---------+---------+
|
||||||
| 8 | 2 | 3 |
|
| 8 | 2 | 4 |
|
||||||
+----------------+---------+---------+
|
+----------------+---------+---------+
|
||||||
| 12 | 3 | 3 |
|
| 12 | 3 | 4 |
|
||||||
+----------------+---------+---------+
|
+----------------+---------+---------+
|
||||||
| 16 | 3 | 5 |
|
| 16 | 4 | 4 |
|
||||||
+----------------+---------+---------+
|
+----------------+---------+---------+
|
||||||
|
|
||||||
If you only specify PAPERLESS_TASK_WORKERS, paperless will adjust
|
If you only specify PAPERLESS_TASK_WORKERS, paperless will adjust
|
||||||
|
147
docs/setup.rst
147
docs/setup.rst
@ -20,45 +20,45 @@ Paperless consists of the following components:
|
|||||||
.. code:: shell-session
|
.. code:: shell-session
|
||||||
|
|
||||||
$ cd /path/to/paperless/src/
|
$ cd /path/to/paperless/src/
|
||||||
$ pipenv run gunicorn -c /usr/src/paperless/gunicorn.conf.py -b 0.0.0.0:8000 paperless.wsgi
|
$ gunicorn -c ../gunicorn.conf.py -b 0.0.0.0:8000 paperless.wsgi
|
||||||
|
|
||||||
or by any other means such as Apache ``mod_wsgi``.
|
or by any other means such as Apache ``mod_wsgi``.
|
||||||
|
|
||||||
* **The consumer:** This is what watches your consumption folder for documents.
|
* **The consumer:** This is what watches your consumption folder for documents.
|
||||||
However, the consumer itself does not consume really consume your documents anymore.
|
However, the consumer itself does not really consume your documents.
|
||||||
It rather notifies a task processor that a new file is ready for consumption.
|
Now it notifies a task processor that a new file is ready for consumption.
|
||||||
I suppose it should be named differently.
|
I suppose it should be named differently.
|
||||||
This also used to check your emails, but that's now gone elsewhere as well.
|
This was also used to check your emails, but that's now done elsewhere as well.
|
||||||
|
|
||||||
Start the consumer with the management command ``document_consumer``:
|
Start the consumer with the management command ``document_consumer``:
|
||||||
|
|
||||||
.. code:: shell-session
|
.. code:: shell-session
|
||||||
|
|
||||||
$ cd /path/to/paperless/src/
|
$ cd /path/to/paperless/src/
|
||||||
$ pipenv run python3 manage.py document_consumer
|
$ python3 manage.py document_consumer
|
||||||
|
|
||||||
.. _setup-task_processor:
|
.. _setup-task_processor:
|
||||||
|
|
||||||
* **The task processor:** Paperless relies on `Django Q <https://django-q.readthedocs.io/en/latest/>`_
|
* **The task processor:** Paperless relies on `Django Q <https://django-q.readthedocs.io/en/latest/>`_
|
||||||
for doing much of the heavy lifting. This is a task queue that accepts tasks from
|
for doing most of the heavy lifting. This is a task queue that accepts tasks from
|
||||||
multiple sources and processes tasks in parallel. It also comes with a scheduler that executes
|
multiple sources and processes these in parallel. It also comes with a scheduler that executes
|
||||||
certain commands periodically.
|
certain commands periodically.
|
||||||
|
|
||||||
This task processor is responsible for:
|
This task processor is responsible for:
|
||||||
|
|
||||||
* Consuming documents. When the consumer finds new documents, it notifies the task processor to
|
* Consuming documents. When the consumer finds new documents, it notifies the task processor to
|
||||||
start a consumption task.
|
start a consumption task.
|
||||||
* Consuming emails. It periodically checks your configured accounts for new mails and
|
|
||||||
produces consumption tasks for any documents it finds.
|
|
||||||
* The task processor also performs the consumption of any documents you upload through
|
* The task processor also performs the consumption of any documents you upload through
|
||||||
the web interface.
|
the web interface.
|
||||||
* Maintain the search index and the automatic matching algorithm. These are things that paperless
|
* Consuming emails. It periodically checks your configured accounts for new emails and
|
||||||
|
notifies the task processor to consume the attachment of an email.
|
||||||
|
* Maintaining the search index and the automatic matching algorithm. These are things that paperless
|
||||||
needs to do from time to time in order to operate properly.
|
needs to do from time to time in order to operate properly.
|
||||||
|
|
||||||
This allows paperless to process multiple documents from your consumption folder in parallel! On
|
This allows paperless to process multiple documents from your consumption folder in parallel! On
|
||||||
a modern multi core system, consumption with full ocr is blazing fast.
|
a modern multi core system, this makes the consumption process with full OCR blazingly fast.
|
||||||
|
|
||||||
The task processor comes with a built-in admin interface that you can use to see whenever any of the
|
The task processor comes with a built-in admin interface that you can use to check whenever any of the
|
||||||
tasks fail and inspect the errors (i.e., wrong email credentials, errors during consuming a specific
|
tasks fail and inspect the errors (i.e., wrong email credentials, errors during consuming a specific
|
||||||
file, etc).
|
file, etc).
|
||||||
|
|
||||||
@ -67,11 +67,11 @@ Paperless consists of the following components:
|
|||||||
.. code:: shell-session
|
.. code:: shell-session
|
||||||
|
|
||||||
$ cd /path/to/paperless/src/
|
$ cd /path/to/paperless/src/
|
||||||
$ pipenv run python3 manage.py qcluster
|
$ python3 manage.py qcluster
|
||||||
|
|
||||||
* A `redis <https://redis.io/>`_ message broker: This is a really lightweight service that is responsible
|
* A `redis <https://redis.io/>`_ message broker: This is a really lightweight service that is responsible
|
||||||
for getting the tasks from the webserver and consumer to the task scheduler. These run in different
|
for getting the tasks from the webserver and the consumer to the task scheduler. These run in a different
|
||||||
processes (maybe even on different machines!), and therefore, this is necessary.
|
process (maybe even on different machines!), and therefore, this is necessary.
|
||||||
|
|
||||||
* Optional: A database server. Paperless supports both PostgreSQL and SQLite for storing its data.
|
* Optional: A database server. Paperless supports both PostgreSQL and SQLite for storing its data.
|
||||||
|
|
||||||
@ -79,7 +79,7 @@ Paperless consists of the following components:
|
|||||||
Installation
|
Installation
|
||||||
############
|
############
|
||||||
|
|
||||||
You can go multiple routes with setting up and running Paperless:
|
You can go multiple routes to setup and run Paperless:
|
||||||
|
|
||||||
* :ref:`Pull the image from Docker Hub <setup-docker_hub>`
|
* :ref:`Pull the image from Docker Hub <setup-docker_hub>`
|
||||||
* :ref:`Build the Docker image yourself <setup-docker_build>`
|
* :ref:`Build the Docker image yourself <setup-docker_build>`
|
||||||
@ -87,26 +87,31 @@ You can go multiple routes with setting up and running Paperless:
|
|||||||
* :ref:`Use ansible to install Paperless on your system automatically (bare metal) <setup-ansible>`
|
* :ref:`Use ansible to install Paperless on your system automatically (bare metal) <setup-ansible>`
|
||||||
|
|
||||||
The Docker routes are quick & easy. These are the recommended routes. This configures all the stuff
|
The Docker routes are quick & easy. These are the recommended routes. This configures all the stuff
|
||||||
from above automatically so that it just works and uses sensible defaults for all configuration options.
|
from the above automatically so that it just works and uses sensible defaults for all configuration options.
|
||||||
|
Here you find a cheat-sheet for docker beginners: `CLI Basics <https://sehn.tech/post/devops-with-docker/>`_
|
||||||
|
|
||||||
The bare metal route is more complicated to setup but makes it easier
|
The bare metal route is complicated to setup but makes it easier
|
||||||
should you want to contribute some code back. You need to configure and
|
should you want to contribute some code back. You need to configure and
|
||||||
run the above mentioned components yourself.
|
run the above mentioned components yourself.
|
||||||
|
|
||||||
The ansible route cobines benefits from both options:
|
The ansible route combines benefits of both options:
|
||||||
the setup process is fully automated, reproducible and idempotent,
|
the setup process is fully automated, reproducible and `idempotent <https://docs.ansible.com/ansible/latest/reference_appendices/glossary.html#Idempotency>`_,
|
||||||
it includes the same sensible defaults,
|
it includes the same sensible defaults, and it simultaneously provides the flexibility of a bare metal installation.
|
||||||
and it simultaneously provides the flexibility of a bare metal installation.
|
|
||||||
|
.. _CLI Basics: https://sehn.tech/post/devops-with-docker/
|
||||||
|
.. _idempotent: https://docs.ansible.com/ansible/latest/reference_appendices/glossary.html#Idempotency
|
||||||
|
|
||||||
.. _setup-docker_hub:
|
.. _setup-docker_hub:
|
||||||
|
|
||||||
Install Paperless from Docker Hub
|
Install Paperless from Docker Hub
|
||||||
=================================
|
=================================
|
||||||
|
|
||||||
1. Go to the `/docker/compose directory on the project page <https://github.com/jonaswinkler/paperless-ng/tree/master/docker/compose>`_
|
1. Login with your user and create a folder in your home-directory `mkdir -v ~/paperless-ng` to have a place for your configuration files and consumption directory.
|
||||||
and download one of the ``docker-compose.*.yml`` files, depending on which database backend you
|
|
||||||
|
2. Go to the `/docker/compose directory on the project page <https://github.com/jonaswinkler/paperless-ng/tree/master/docker/compose>`_
|
||||||
|
and download one of the `docker-compose.*.yml` files, depending on which database backend you
|
||||||
want to use. Rename this file to `docker-compose.yml`.
|
want to use. Rename this file to `docker-compose.yml`.
|
||||||
If you want to enable optional support for Office documents, download a file with ``-tika`` in its name.
|
If you want to enable optional support for Office documents, download a file with `-tika` in the file name.
|
||||||
Download the ``docker-compose.env`` file and the ``.env`` file as well and store them
|
Download the ``docker-compose.env`` file and the ``.env`` file as well and store them
|
||||||
in the same directory.
|
in the same directory.
|
||||||
|
|
||||||
@ -115,25 +120,26 @@ Install Paperless from Docker Hub
|
|||||||
For new installations, it is recommended to use PostgreSQL as the database
|
For new installations, it is recommended to use PostgreSQL as the database
|
||||||
backend.
|
backend.
|
||||||
|
|
||||||
2. Install `Docker`_ and `docker-compose`_.
|
3. Install `Docker`_ and `docker-compose`_.
|
||||||
|
|
||||||
.. caution::
|
.. caution::
|
||||||
|
|
||||||
If you want to use the included ``docker-compose.*.yml`` file, you
|
If you want to use the included ``docker-compose.*.yml`` file, you
|
||||||
need to have at least Docker version **17.09.0** and docker-compose
|
need to have at least Docker version **17.09.0** and docker-compose
|
||||||
version **1.17.0**.
|
version **1.17.0**.
|
||||||
|
To check do: `docker-compose -v` or `docker -v`
|
||||||
|
|
||||||
See the `Docker installation guide`_ on how to install the current
|
See the `Docker installation guide`_ on how to install the current
|
||||||
version of Docker for your operating system or Linux distribution of
|
version of Docker for your operating system or Linux distribution of
|
||||||
choice. To get an up-to-date version of docker-compose, follow the
|
choice. To get the latest version of docker-compose, follow the
|
||||||
`docker-compose installation guide`_ if your package repository doesn't
|
`docker-compose installation guide`_ if your package repository doesn't
|
||||||
include it.
|
include it.
|
||||||
|
|
||||||
.. _Docker installation guide: https://docs.docker.com/engine/installation/
|
.. _Docker installation guide: https://docs.docker.com/engine/installation/
|
||||||
.. _docker-compose installation guide: https://docs.docker.com/compose/install/
|
.. _docker-compose installation guide: https://docs.docker.com/compose/install/
|
||||||
|
|
||||||
3. Modify ``docker-compose.yml`` to your preferences. You may want to change the path
|
4. Modify ``docker-compose.yml`` to your preferences. You may want to change the path
|
||||||
to the consumption directory in this file. Find the line that specifies where
|
to the consumption directory. Find the line that specifies where
|
||||||
to mount the consumption directory:
|
to mount the consumption directory:
|
||||||
|
|
||||||
.. code::
|
.. code::
|
||||||
@ -149,31 +155,35 @@ Install Paperless from Docker Hub
|
|||||||
Don't change the part after the colon or paperless wont find your documents.
|
Don't change the part after the colon or paperless wont find your documents.
|
||||||
|
|
||||||
|
|
||||||
4. Modify ``docker-compose.env``, following the comments in the file. The
|
5. Modify ``docker-compose.env``, following the comments in the file. The
|
||||||
most important change is to set ``USERMAP_UID`` and ``USERMAP_GID``
|
most important change is to set ``USERMAP_UID`` and ``USERMAP_GID``
|
||||||
to the uid and gid of your user on the host system. This ensures that
|
to the uid and gid of your user on the host system. This ensures that
|
||||||
both the docker container and you on the host machine have write access
|
both the docker container and you on the host machine have write access
|
||||||
to the consumption directory. If your UID and GID on the host system is
|
to the consumption directory. If your UID and GID on the host system is
|
||||||
1000 (the default for the first normal user on most systems), it will
|
1000 (the default for the first normal user on most systems), it will
|
||||||
work out of the box without any modifications.
|
work out of the box without any modifications. `id "username"` to check.
|
||||||
|
|
||||||
.. note::
|
.. note::
|
||||||
|
|
||||||
You can use any settings from the file ``paperless.conf.example`` in this file.
|
You can copy any setting from the file ``paperless.conf.example`` and paste it here.
|
||||||
Have a look at :ref:`configuration` to see whats available.
|
Have a look at :ref:`configuration` to see what's available.
|
||||||
|
|
||||||
.. caution::
|
.. caution::
|
||||||
|
|
||||||
Certain file systems such as NFS network shares don't support file system
|
Some file systems such as NFS network shares don't support file system
|
||||||
notifications with ``inotify``. When storing the consumption directory
|
notifications with ``inotify``. When storing the consumption directory
|
||||||
on such a file system, paperless will be unable to pick up new files
|
on such a file system, paperless will not pick up new files
|
||||||
with the default configuration. You will need to use ``PAPERLESS_CONSUMER_POLLING``,
|
with the default configuration. You will need to use ``PAPERLESS_CONSUMER_POLLING``,
|
||||||
which will disable inotify. See :ref:`here <configuration-polling>`.
|
which will disable inotify. See :ref:`here <configuration-polling>`.
|
||||||
|
|
||||||
5. Run ``docker-compose up -d``. This will create and start the necessary
|
6. Now head over to: https://hub.docker.com/r/jonaswinkler/paperless-ng and choose your preferred
|
||||||
containers.
|
image and copy the link. To download this image do a `docker pull` followed by the link. Do this within the directory with the .yml files.
|
||||||
|
Depending on your network connection and CPU this will take a while. You have time to get a beverage.
|
||||||
|
|
||||||
6. To be able to login, you will need a super user. To create it, execute the
|
7. Run ``docker-compose up -d``. This will create and start the necessary
|
||||||
|
containers, but your are not done yet!
|
||||||
|
|
||||||
|
8. To be able to login, you will need a super user. To create it, execute the
|
||||||
following command:
|
following command:
|
||||||
|
|
||||||
.. code-block:: shell-session
|
.. code-block:: shell-session
|
||||||
@ -181,12 +191,12 @@ Install Paperless from Docker Hub
|
|||||||
$ docker-compose run --rm webserver createsuperuser
|
$ docker-compose run --rm webserver createsuperuser
|
||||||
|
|
||||||
This will prompt you to set a username, an optional e-mail address and
|
This will prompt you to set a username, an optional e-mail address and
|
||||||
finally a password.
|
finally a password (at least 8 characters).
|
||||||
|
|
||||||
7. The default ``docker-compose.yml`` exports the webserver on your local port
|
9. The default ``docker-compose.yml`` exports the webserver on your local port
|
||||||
8000. If you haven't adapted this, you should now be able to visit your
|
8000. If you haven't adapted this, you should now be able to visit your
|
||||||
Paperless instance at ``http://127.0.0.1:8000``. You can login with the
|
Paperless instance at ``http://127.0.0.1:8000`` or your servers IP-Address:8000.
|
||||||
user and password you just created.
|
Use the login credentials you have created with the previous step.
|
||||||
|
|
||||||
.. _Docker: https://www.docker.com/
|
.. _Docker: https://www.docker.com/
|
||||||
.. _docker-compose: https://docs.docker.com/compose/install/
|
.. _docker-compose: https://docs.docker.com/compose/install/
|
||||||
@ -214,7 +224,7 @@ Build the docker image yourself
|
|||||||
|
|
||||||
webserver:
|
webserver:
|
||||||
image: jonaswinkler/paperless-ng:latest
|
image: jonaswinkler/paperless-ng:latest
|
||||||
|
|
||||||
and replace it with a line that instructs docker-compose to build the image from the current working directory instead:
|
and replace it with a line that instructs docker-compose to build the image from the current working directory instead:
|
||||||
|
|
||||||
.. code:: yaml
|
.. code:: yaml
|
||||||
@ -245,7 +255,7 @@ writing. Windows is not and will never be supported.
|
|||||||
1. Install dependencies. Paperless requires the following packages.
|
1. Install dependencies. Paperless requires the following packages.
|
||||||
|
|
||||||
* ``python3`` 3.6, 3.7, 3.8, 3.9
|
* ``python3`` 3.6, 3.7, 3.8, 3.9
|
||||||
* ``python3-pip``, optionally ``pipenv`` for package installation
|
* ``python3-pip``
|
||||||
* ``python3-dev``
|
* ``python3-dev``
|
||||||
|
|
||||||
* ``fonts-liberation`` for generating thumbnails for plain text files
|
* ``fonts-liberation`` for generating thumbnails for plain text files
|
||||||
@ -314,8 +324,13 @@ writing. Windows is not and will never be supported.
|
|||||||
|
|
||||||
Adjust as necessary if you configured different folders.
|
Adjust as necessary if you configured different folders.
|
||||||
|
|
||||||
7. Install python requirements. Paperless comes with both Pipfiles for ``pipenv`` as well as with a ``requirements.txt``.
|
7. Install python requirements from the ``requirements.txt`` file.
|
||||||
Both will install exactly the same requirements. It is up to you if you wish to use a virtual environment or not.
|
It is up to you if you wish to use a virtual environment or not.
|
||||||
|
|
||||||
|
.. code:: shell-session
|
||||||
|
|
||||||
|
pip3 install -r requirements.txt
|
||||||
|
|
||||||
|
|
||||||
8. Go to ``/opt/paperless/src``, and execute the following commands:
|
8. Go to ``/opt/paperless/src``, and execute the following commands:
|
||||||
|
|
||||||
@ -339,7 +354,8 @@ writing. Windows is not and will never be supported.
|
|||||||
.. warning::
|
.. warning::
|
||||||
|
|
||||||
This is a development server which should not be used in
|
This is a development server which should not be used in
|
||||||
production.
|
production. It is not audited for security and performance
|
||||||
|
is inferior to production ready web servers.
|
||||||
|
|
||||||
.. hint::
|
.. hint::
|
||||||
|
|
||||||
@ -354,6 +370,11 @@ writing. Windows is not and will never be supported.
|
|||||||
``consumer`` script to watch the input folder, and the ``scheduler``
|
``consumer`` script to watch the input folder, and the ``scheduler``
|
||||||
script to run tasks such as email checking and document consumption.
|
script to run tasks such as email checking and document consumption.
|
||||||
|
|
||||||
|
You may need to adjust the path to the ``gunicorn`` executable. This
|
||||||
|
will be installed as part of the python dependencies, and is either located
|
||||||
|
in the ``bin`` folder of your virtual environment, or in ``~/.local/bin/`` if
|
||||||
|
no virtual environment is used.
|
||||||
|
|
||||||
These services rely on redis and optionally the database server, but
|
These services rely on redis and optionally the database server, but
|
||||||
don't need to be started in any particular order. The example files
|
don't need to be started in any particular order. The example files
|
||||||
depend on redis being started. If you use a database server, you should
|
depend on redis being started. If you use a database server, you should
|
||||||
@ -406,7 +427,7 @@ Install Paperless using ansible
|
|||||||
|
|
||||||
This role currently only supports Debian 10 Buster and Ubuntu 20.04 Focal or later as target hosts.
|
This role currently only supports Debian 10 Buster and Ubuntu 20.04 Focal or later as target hosts.
|
||||||
|
|
||||||
1. Install ansible 2.7+ on the management node.
|
1. Install ansible 2.7+ on the management node.
|
||||||
This may be the target host paperless-ng is being installed on or any remote host which can access the target host.
|
This may be the target host paperless-ng is being installed on or any remote host which can access the target host.
|
||||||
For further details, check the ansible `inventory <https://docs.ansible.com/ansible/latest/user_guide/intro_inventory.html>`_ documentation.
|
For further details, check the ansible `inventory <https://docs.ansible.com/ansible/latest/user_guide/intro_inventory.html>`_ documentation.
|
||||||
|
|
||||||
@ -518,7 +539,10 @@ Migration to paperless-ng
|
|||||||
|
|
||||||
At its core, paperless-ng is still paperless and fully compatible. However, some
|
At its core, paperless-ng is still paperless and fully compatible. However, some
|
||||||
things have changed under the hood, so you need to adapt your setup depending on
|
things have changed under the hood, so you need to adapt your setup depending on
|
||||||
how you installed paperless. The important things to keep in mind are as follows.
|
how you installed paperless.
|
||||||
|
|
||||||
|
This setup describes how to update an existing paperless Docker installation.
|
||||||
|
The important things to keep in mind are as follows:
|
||||||
|
|
||||||
* Read the :ref:`changelog <paperless_changelog>` and take note of breaking changes.
|
* Read the :ref:`changelog <paperless_changelog>` and take note of breaking changes.
|
||||||
* You should decide if you want to stick with SQLite or want to migrate your database
|
* You should decide if you want to stick with SQLite or want to migrate your database
|
||||||
@ -553,11 +577,18 @@ Migration to paperless-ng is then performed in a few simple steps:
|
|||||||
|
|
||||||
.. caution::
|
.. caution::
|
||||||
|
|
||||||
Paperless includes a ``.env`` file. This will set the
|
Paperless-ng includes a ``.env`` file. This will set the
|
||||||
project name for docker compose to ``paperless`` so that paperless-ng will
|
project name for docker compose to ``paperless``, which will also define the name
|
||||||
automatically reuse your existing paperless volumes. When you start it, it
|
of the volumes by paperless-ng. However, if you experience that paperless-ng
|
||||||
will migrate your existing data. After that, your old paperless installation
|
is not using your old paperless volumes, verify the names of your volumes with
|
||||||
will be incompatible with the migrated volumes.
|
|
||||||
|
.. code:: shell-session
|
||||||
|
|
||||||
|
$ docker volume ls | grep _data
|
||||||
|
|
||||||
|
and adjust the project name in the ``.env`` file so that it matches the name
|
||||||
|
of the volumes before the ``_data`` part.
|
||||||
|
|
||||||
|
|
||||||
4. Download the ``docker-compose.sqlite.yml`` file to ``docker-compose.yml``.
|
4. Download the ``docker-compose.sqlite.yml`` file to ``docker-compose.yml``.
|
||||||
If you want to switch to PostgreSQL, do that after you migrated your existing
|
If you want to switch to PostgreSQL, do that after you migrated your existing
|
||||||
@ -638,14 +669,12 @@ management commands as below.
|
|||||||
|
|
||||||
This will launch the container and initialize the PostgreSQL database.
|
This will launch the container and initialize the PostgreSQL database.
|
||||||
|
|
||||||
b) Without docker, open a shell in your virtual environment, switch to
|
b) Without docker, remember to activate any virtual environment, switch to
|
||||||
the ``src`` directory and create the database schema:
|
the ``src`` directory and create the database schema:
|
||||||
|
|
||||||
.. code:: shell-session
|
.. code:: shell-session
|
||||||
|
|
||||||
$ cd /path/to/paperless
|
$ cd /path/to/paperless/src
|
||||||
$ pipenv shell
|
|
||||||
$ cd src
|
|
||||||
$ python3 manage.py migrate
|
$ python3 manage.py migrate
|
||||||
|
|
||||||
This will not copy any data yet.
|
This will not copy any data yet.
|
||||||
@ -662,7 +691,7 @@ management commands as below.
|
|||||||
|
|
||||||
$ python3 manage.py loaddata data.json
|
$ python3 manage.py loaddata data.json
|
||||||
|
|
||||||
6. Exit the shell.
|
6. If operating inside Docker, you may exit the shell now.
|
||||||
|
|
||||||
.. code:: shell-session
|
.. code:: shell-session
|
||||||
|
|
||||||
|
@ -30,13 +30,22 @@ Consumer fails to pickup any new files
|
|||||||
######################################
|
######################################
|
||||||
|
|
||||||
If you notice that the consumer will only pickup files in the consumption
|
If you notice that the consumer will only pickup files in the consumption
|
||||||
directory at startup, but won't find any other files added later, check out
|
directory at startup, but won't find any other files added later, you will need to
|
||||||
the configuration file and enable filesystem polling with the setting
|
enable filesystem polling with the configuration option
|
||||||
``PAPERLESS_CONSUMER_POLLING``.
|
``PAPERLESS_CONSUMER_POLLING``, see :ref:`here <configuration-polling>`.
|
||||||
|
|
||||||
This will disable listening to filesystem changes with inotify and paperless will
|
This will disable listening to filesystem changes with inotify and paperless will
|
||||||
manually check the consumption directory for changes instead.
|
manually check the consumption directory for changes instead.
|
||||||
|
|
||||||
|
|
||||||
|
Paperless always redirects to /admin
|
||||||
|
####################################
|
||||||
|
|
||||||
|
You probably had the old paperless installed at some point. Paperless installed
|
||||||
|
a permanent redirect to /admin in your browser, and you need to clear your
|
||||||
|
browsing data / cache to fix that.
|
||||||
|
|
||||||
|
|
||||||
Operation not permitted
|
Operation not permitted
|
||||||
#######################
|
#######################
|
||||||
|
|
||||||
@ -64,6 +73,24 @@ This may have two reasons:
|
|||||||
with Inbox tags. Verify that there are documents in your archive without inbox tags.
|
with Inbox tags. Verify that there are documents in your archive without inbox tags.
|
||||||
The algorithm will only learn from documents not in your inbox.
|
The algorithm will only learn from documents not in your inbox.
|
||||||
|
|
||||||
|
UserWarning in sklearn on every single document
|
||||||
|
###############################################
|
||||||
|
|
||||||
|
You may encounter warnings like this:
|
||||||
|
|
||||||
|
.. code::
|
||||||
|
|
||||||
|
/usr/local/lib/python3.7/site-packages/sklearn/base.py:315:
|
||||||
|
UserWarning: Trying to unpickle estimator CountVectorizer from version 0.23.2 when using version 0.24.0.
|
||||||
|
This might lead to breaking code or invalid results. Use at your own risk.
|
||||||
|
|
||||||
|
This happens when certain dependencies of paperless that are responsible for the auto matching algorithm are
|
||||||
|
updated. After updating these, your current training data *might* not be compatible anymore. This can be ignored
|
||||||
|
in most cases. This warning will disappear automatically when paperless updates the training data.
|
||||||
|
|
||||||
|
If you want to get rid of the warning or actually experience issues with automatic matching, delete
|
||||||
|
the file ``classification_model.pickle`` in the data directory and let paperless recreate it.
|
||||||
|
|
||||||
Permission denied errors in the consumption directory
|
Permission denied errors in the consumption directory
|
||||||
#####################################################
|
#####################################################
|
||||||
|
|
||||||
@ -78,3 +105,47 @@ Ensure that ``USERMAP_UID`` and ``USERMAP_GID`` are set to the user id and group
|
|||||||
different from ``1000``. See :ref:`setup-docker_hub`.
|
different from ``1000``. See :ref:`setup-docker_hub`.
|
||||||
|
|
||||||
Also ensure that you are able to read and write to the consumption directory on the host.
|
Also ensure that you are able to read and write to the consumption directory on the host.
|
||||||
|
|
||||||
|
Web-UI stuck at "Loading..."
|
||||||
|
############################
|
||||||
|
|
||||||
|
This might have multiple reasons.
|
||||||
|
|
||||||
|
|
||||||
|
1. If you built the docker image yourself or deployed using the bare metal route,
|
||||||
|
make sure that there are files in ``<paperless-root>/static/frontend/<lang-code>/``.
|
||||||
|
If there are no files, make sure that you executed ``collectstatic`` successfully, either
|
||||||
|
manually or as part of the docker image build.
|
||||||
|
|
||||||
|
If the front end is still missing, make sure that the front end is compiled (files present in
|
||||||
|
``src/documents/static/frontend``). If it is not, you need to compile the front end yourself
|
||||||
|
or download the release archive instead of cloning the repository.
|
||||||
|
|
||||||
|
2. Check the output of the web server. You might see errors like this:
|
||||||
|
|
||||||
|
|
||||||
|
.. code::
|
||||||
|
|
||||||
|
[2021-01-25 10:08:04 +0000] [40] [ERROR] Socket error processing request.
|
||||||
|
Traceback (most recent call last):
|
||||||
|
File "/usr/local/lib/python3.7/site-packages/gunicorn/workers/sync.py", line 134, in handle
|
||||||
|
self.handle_request(listener, req, client, addr)
|
||||||
|
File "/usr/local/lib/python3.7/site-packages/gunicorn/workers/sync.py", line 190, in handle_request
|
||||||
|
util.reraise(*sys.exc_info())
|
||||||
|
File "/usr/local/lib/python3.7/site-packages/gunicorn/util.py", line 625, in reraise
|
||||||
|
raise value
|
||||||
|
File "/usr/local/lib/python3.7/site-packages/gunicorn/workers/sync.py", line 178, in handle_request
|
||||||
|
resp.write_file(respiter)
|
||||||
|
File "/usr/local/lib/python3.7/site-packages/gunicorn/http/wsgi.py", line 396, in write_file
|
||||||
|
if not self.sendfile(respiter):
|
||||||
|
File "/usr/local/lib/python3.7/site-packages/gunicorn/http/wsgi.py", line 386, in sendfile
|
||||||
|
sent += os.sendfile(sockno, fileno, offset + sent, count)
|
||||||
|
OSError: [Errno 22] Invalid argument
|
||||||
|
|
||||||
|
To fix this issue, add
|
||||||
|
|
||||||
|
.. code::
|
||||||
|
|
||||||
|
SENDFILE=0
|
||||||
|
|
||||||
|
to your `docker-compose.env` file.
|
@ -188,35 +188,35 @@
|
|||||||
<source>Confirm delete</source>
|
<source>Confirm delete</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
||||||
<context context-type="linenumber">192</context>
|
<context context-type="linenumber">199</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="5382975254277698192" datatype="html">
|
<trans-unit id="5382975254277698192" datatype="html">
|
||||||
<source>Do you really want to delete document "<x id="PH" equiv-text="this.document.title"/>"?</source>
|
<source>Do you really want to delete document "<x id="PH" equiv-text="this.document.title"/>"?</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
||||||
<context context-type="linenumber">193</context>
|
<context context-type="linenumber">200</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="6691075929777935948" datatype="html">
|
<trans-unit id="6691075929777935948" datatype="html">
|
||||||
<source>The files for this document will be deleted permanently. This operation cannot be undone.</source>
|
<source>The files for this document will be deleted permanently. This operation cannot be undone.</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
||||||
<context context-type="linenumber">194</context>
|
<context context-type="linenumber">201</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="719892092227206532" datatype="html">
|
<trans-unit id="719892092227206532" datatype="html">
|
||||||
<source>Delete document</source>
|
<source>Delete document</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
||||||
<context context-type="linenumber">196</context>
|
<context context-type="linenumber">203</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="1844801255494293730" datatype="html">
|
<trans-unit id="1844801255494293730" datatype="html">
|
||||||
<source>Error deleting document: <x id="PH" equiv-text="JSON.stringify(error)"/></source>
|
<source>Error deleting document: <x id="PH" equiv-text="JSON.stringify(error)"/></source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
||||||
<context context-type="linenumber">203</context>
|
<context context-type="linenumber">210</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="826b25211922a1b46436589233cb6f1a163d89b7" datatype="html">
|
<trans-unit id="826b25211922a1b46436589233cb6f1a163d89b7" datatype="html">
|
||||||
@ -1215,6 +1215,13 @@
|
|||||||
<context context-type="linenumber">73</context>
|
<context context-type="linenumber">73</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
|
<trans-unit id="7894972847287473517" datatype="html">
|
||||||
|
<source>"<x id="PH" equiv-text="items[0].name"/>"</source>
|
||||||
|
<context-group purpose="location">
|
||||||
|
<context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.ts</context>
|
||||||
|
<context context-type="linenumber">112</context>
|
||||||
|
</context-group>
|
||||||
|
</trans-unit>
|
||||||
<trans-unit id="8639884465898458690" datatype="html">
|
<trans-unit id="8639884465898458690" datatype="html">
|
||||||
<source>"<x id="PH" equiv-text="items[0].name"/>" and "<x id="PH_1" equiv-text="items[1].name"/>"</source>
|
<source>"<x id="PH" equiv-text="items[0].name"/>" and "<x id="PH_1" equiv-text="items[1].name"/>"</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
@ -1223,13 +1230,6 @@
|
|||||||
</context-group>
|
</context-group>
|
||||||
<note priority="1" from="description">This is for messages like 'modify "tag1" and "tag2"'</note>
|
<note priority="1" from="description">This is for messages like 'modify "tag1" and "tag2"'</note>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="7894972847287473517" datatype="html">
|
|
||||||
<source>"<x id="PH" equiv-text="i.name"/>"</source>
|
|
||||||
<context-group purpose="location">
|
|
||||||
<context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.ts</context>
|
|
||||||
<context context-type="linenumber">116</context>
|
|
||||||
</context-group>
|
|
||||||
</trans-unit>
|
|
||||||
<trans-unit id="760986369763309193" datatype="html">
|
<trans-unit id="760986369763309193" datatype="html">
|
||||||
<source>, </source>
|
<source>, </source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
@ -1379,6 +1379,13 @@
|
|||||||
<context context-type="linenumber">27</context>
|
<context context-type="linenumber">27</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
|
<trans-unit id="a1e6c11f20d4bf6e8e6b43e3c6d2561b2080645e" datatype="html">
|
||||||
|
<source>Suggestions:</source>
|
||||||
|
<context-group purpose="location">
|
||||||
|
<context context-type="sourcefile">src/app/components/common/input/select/select.component.html</context>
|
||||||
|
<context context-type="linenumber">26</context>
|
||||||
|
</context-group>
|
||||||
|
</trans-unit>
|
||||||
<trans-unit id="27d158b47717ff9305d19866960418c603f19d55" datatype="html">
|
<trans-unit id="27d158b47717ff9305d19866960418c603f19d55" datatype="html">
|
||||||
<source>Save current view</source>
|
<source>Save current view</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
@ -1726,49 +1733,49 @@
|
|||||||
<source>ASN</source>
|
<source>ASN</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">16</context>
|
<context context-type="linenumber">17</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="2691296884221415710" datatype="html">
|
<trans-unit id="2691296884221415710" datatype="html">
|
||||||
<source>Correspondent</source>
|
<source>Correspondent</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">17</context>
|
<context context-type="linenumber">18</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="5701618810648052610" datatype="html">
|
<trans-unit id="5701618810648052610" datatype="html">
|
||||||
<source>Title</source>
|
<source>Title</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">18</context>
|
<context context-type="linenumber">19</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="5066119607229701477" datatype="html">
|
<trans-unit id="5066119607229701477" datatype="html">
|
||||||
<source>Document type</source>
|
<source>Document type</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">19</context>
|
<context context-type="linenumber">20</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="4207916966377787111" datatype="html">
|
<trans-unit id="4207916966377787111" datatype="html">
|
||||||
<source>Created</source>
|
<source>Created</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">20</context>
|
<context context-type="linenumber">21</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="231679111972850796" datatype="html">
|
<trans-unit id="231679111972850796" datatype="html">
|
||||||
<source>Added</source>
|
<source>Added</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">21</context>
|
<context context-type="linenumber">22</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="3553216189604488439" datatype="html">
|
<trans-unit id="3553216189604488439" datatype="html">
|
||||||
<source>Modified</source>
|
<source>Modified</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">22</context>
|
<context context-type="linenumber">23</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="2056433880533904076" datatype="html">
|
<trans-unit id="2056433880533904076" datatype="html">
|
||||||
|
@ -22,4 +22,12 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<small *ngIf="hint" class="form-text text-muted">{{hint}}</small>
|
<small *ngIf="hint" class="form-text text-muted">{{hint}}</small>
|
||||||
|
<small *ngIf="getSuggestions().length > 0">
|
||||||
|
<span i18n>Suggestions:</span>
|
||||||
|
<ng-container *ngFor="let s of getSuggestions()">
|
||||||
|
<a (click)="value = s.id; onChange(value)" [routerLink]="">{{s.name}}</a>
|
||||||
|
</ng-container>
|
||||||
|
|
||||||
|
|
||||||
|
</small>
|
||||||
</div>
|
</div>
|
||||||
|
@ -30,11 +30,22 @@ export class SelectComponent extends AbstractInputComponent<number> {
|
|||||||
@Input()
|
@Input()
|
||||||
allowNull: boolean = false
|
allowNull: boolean = false
|
||||||
|
|
||||||
|
@Input()
|
||||||
|
suggestions: number[]
|
||||||
|
|
||||||
@Output()
|
@Output()
|
||||||
createNew = new EventEmitter()
|
createNew = new EventEmitter()
|
||||||
|
|
||||||
showPlusButton(): boolean {
|
showPlusButton(): boolean {
|
||||||
return this.createNew.observers.length > 0
|
return this.createNew.observers.length > 0
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getSuggestions() {
|
||||||
|
if (this.suggestions && this.items) {
|
||||||
|
return this.suggestions.filter(id => id != this.value).map(id => this.items.find(item => item.id == id))
|
||||||
|
} else {
|
||||||
|
return []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -2,30 +2,25 @@
|
|||||||
<label for="tags" i18n>Tags</label>
|
<label for="tags" i18n>Tags</label>
|
||||||
|
|
||||||
<div class="input-group flex-nowrap">
|
<div class="input-group flex-nowrap">
|
||||||
<ng-select name="tags" [items]="tags" bindLabel="name" bindValue="id" [(ngModel)]="displayValue"
|
<ng-select name="tags" [items]="tags" bindLabel="name" bindValue="id" [(ngModel)]="value"
|
||||||
[multiple]="true"
|
[multiple]="true"
|
||||||
[closeOnSelect]="false"
|
[closeOnSelect]="false"
|
||||||
[clearSearchOnAdd]="true"
|
[clearSearchOnAdd]="true"
|
||||||
[disabled]="disabled"
|
|
||||||
[hideSelected]="true"
|
[hideSelected]="true"
|
||||||
(change)="ngSelectChange()">
|
(change)="onChange(value)"
|
||||||
|
(blur)="onTouched()">
|
||||||
|
|
||||||
<ng-template ng-label-tmp let-item="item">
|
<ng-template ng-label-tmp let-item="item">
|
||||||
<span class="tag-wrap tag-wrap-delete" (click)="removeTag(item.id)">
|
<span class="tag-wrap tag-wrap-delete" (click)="removeTag(item.id)">
|
||||||
<svg width="1.2em" height="1em" viewBox="0 0 16 16" fill="currentColor" xmlns="http://www.w3.org/2000/svg">
|
<svg width="1.2em" height="1em" viewBox="0 0 16 16" fill="currentColor" xmlns="http://www.w3.org/2000/svg">
|
||||||
<use xlink:href="assets/bootstrap-icons.svg#x"/>
|
<use xlink:href="assets/bootstrap-icons.svg#x"/>
|
||||||
</svg>
|
</svg>
|
||||||
<app-tag style="background-color: none;" [tag]="getTag(item.id)"></app-tag>
|
<app-tag *ngIf="item.id && tags" style="background-color: none;" [tag]="getTag(item.id)"></app-tag>
|
||||||
</span>
|
</span>
|
||||||
</ng-template>
|
</ng-template>
|
||||||
<ng-template ng-option-tmp let-item="item" let-index="index" let-search="searchTerm">
|
<ng-template ng-option-tmp let-item="item" let-index="index" let-search="searchTerm">
|
||||||
<div class="tag-wrap">
|
<div class="tag-wrap">
|
||||||
<div class="selected-icon d-inline-block mr-1">
|
<app-tag *ngIf="item.id && tags" class="mr-2" [tag]="getTag(item.id)"></app-tag>
|
||||||
<svg *ngIf="displayValue.includes(item.id)" width="1em" height="1em" viewBox="0 0 16 16" fill="currentColor" xmlns="http://www.w3.org/2000/svg">
|
|
||||||
<use xlink:href="assets/bootstrap-icons.svg#check"/>
|
|
||||||
</svg>
|
|
||||||
</div>
|
|
||||||
<app-tag class="mr-2" [tag]="getTag(item.id)"></app-tag>
|
|
||||||
</div>
|
</div>
|
||||||
</ng-template>
|
</ng-template>
|
||||||
</ng-select>
|
</ng-select>
|
||||||
@ -39,5 +34,13 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<small class="form-text text-muted" *ngIf="hint">{{hint}}</small>
|
<small class="form-text text-muted" *ngIf="hint">{{hint}}</small>
|
||||||
|
<small *ngIf="getSuggestions().length > 0">
|
||||||
|
<span i18n>Suggestions:</span>
|
||||||
|
<ng-container *ngFor="let tag of getSuggestions()">
|
||||||
|
<a (click)="addTag(tag.id)" [routerLink]="">{{tag.name}}</a>
|
||||||
|
</ng-container>
|
||||||
|
|
||||||
|
|
||||||
|
</small>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
@ -26,9 +26,6 @@ export class TagsComponent implements OnInit, ControlValueAccessor {
|
|||||||
|
|
||||||
writeValue(newValue: number[]): void {
|
writeValue(newValue: number[]): void {
|
||||||
this.value = newValue
|
this.value = newValue
|
||||||
if (this.tags) {
|
|
||||||
this.displayValue = newValue
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
registerOnChange(fn: any): void {
|
registerOnChange(fn: any): void {
|
||||||
this.onChange = fn;
|
this.onChange = fn;
|
||||||
@ -43,7 +40,6 @@ export class TagsComponent implements OnInit, ControlValueAccessor {
|
|||||||
ngOnInit(): void {
|
ngOnInit(): void {
|
||||||
this.tagService.listAll().subscribe(result => {
|
this.tagService.listAll().subscribe(result => {
|
||||||
this.tags = result.results
|
this.tags = result.results
|
||||||
this.displayValue = this.value
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -53,23 +49,28 @@ export class TagsComponent implements OnInit, ControlValueAccessor {
|
|||||||
@Input()
|
@Input()
|
||||||
hint
|
hint
|
||||||
|
|
||||||
value: number[]
|
@Input()
|
||||||
|
suggestions: number[]
|
||||||
|
|
||||||
displayValue: number[] = []
|
value: number[]
|
||||||
|
|
||||||
tags: PaperlessTag[]
|
tags: PaperlessTag[]
|
||||||
|
|
||||||
getTag(id) {
|
getTag(id) {
|
||||||
return this.tags.find(tag => tag.id == id)
|
if (this.tags) {
|
||||||
|
return this.tags.find(tag => tag.id == id)
|
||||||
|
} else {
|
||||||
|
return null
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
removeTag(id) {
|
removeTag(id) {
|
||||||
let index = this.displayValue.indexOf(id)
|
let index = this.value.indexOf(id)
|
||||||
if (index > -1) {
|
if (index > -1) {
|
||||||
let oldValue = this.displayValue
|
let oldValue = this.value
|
||||||
oldValue.splice(index, 1)
|
oldValue.splice(index, 1)
|
||||||
this.displayValue = [...oldValue]
|
this.value = [...oldValue]
|
||||||
this.onChange(this.displayValue)
|
this.onChange(this.value)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -79,15 +80,23 @@ export class TagsComponent implements OnInit, ControlValueAccessor {
|
|||||||
modal.componentInstance.success.subscribe(newTag => {
|
modal.componentInstance.success.subscribe(newTag => {
|
||||||
this.tagService.listAll().subscribe(tags => {
|
this.tagService.listAll().subscribe(tags => {
|
||||||
this.tags = tags.results
|
this.tags = tags.results
|
||||||
this.displayValue = [...this.displayValue, newTag.id]
|
this.value = [...this.value, newTag.id]
|
||||||
this.onChange(this.displayValue)
|
this.onChange(this.value)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
ngSelectChange() {
|
getSuggestions() {
|
||||||
this.value = this.displayValue
|
if (this.suggestions && this.tags) {
|
||||||
this.onChange(this.displayValue)
|
return this.suggestions.filter(id => !this.value.includes(id)).map(id => this.tags.find(tag => tag.id == id))
|
||||||
|
} else {
|
||||||
|
return []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
addTag(id) {
|
||||||
|
this.value = [...this.value, id]
|
||||||
|
this.onChange(this.value)
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -60,10 +60,10 @@
|
|||||||
<app-input-number i18n-title title="Archive serial number" [error]="error?.archive_serial_number" formControlName='archive_serial_number'></app-input-number>
|
<app-input-number i18n-title title="Archive serial number" [error]="error?.archive_serial_number" formControlName='archive_serial_number'></app-input-number>
|
||||||
<app-input-date-time i18n-titleDate titleDate="Date created" formControlName="created"></app-input-date-time>
|
<app-input-date-time i18n-titleDate titleDate="Date created" formControlName="created"></app-input-date-time>
|
||||||
<app-input-select [items]="correspondents" i18n-title title="Correspondent" formControlName="correspondent" [allowNull]="true"
|
<app-input-select [items]="correspondents" i18n-title title="Correspondent" formControlName="correspondent" [allowNull]="true"
|
||||||
(createNew)="createCorrespondent()"></app-input-select>
|
(createNew)="createCorrespondent()" [suggestions]="suggestions?.correspondents"></app-input-select>
|
||||||
<app-input-select [items]="documentTypes" i18n-title title="Document type" formControlName="document_type" [allowNull]="true"
|
<app-input-select [items]="documentTypes" i18n-title title="Document type" formControlName="document_type" [allowNull]="true"
|
||||||
(createNew)="createDocumentType()"></app-input-select>
|
(createNew)="createDocumentType()" [suggestions]="suggestions?.document_types"></app-input-select>
|
||||||
<app-input-tags formControlName="tags"></app-input-tags>
|
<app-input-tags formControlName="tags" [suggestions]="suggestions?.tags"></app-input-tags>
|
||||||
|
|
||||||
</ng-template>
|
</ng-template>
|
||||||
</li>
|
</li>
|
||||||
@ -145,6 +145,6 @@
|
|||||||
<ng-container *ngIf="getContentType() == 'text/plain'">
|
<ng-container *ngIf="getContentType() == 'text/plain'">
|
||||||
<object [data]="previewUrl | safe" type="text/plain" class="preview-sticky" width="100%"></object>
|
<object [data]="previewUrl | safe" type="text/plain" class="preview-sticky" width="100%"></object>
|
||||||
</ng-container>
|
</ng-container>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -19,6 +19,7 @@ import { PDFDocumentProxy } from 'ng2-pdf-viewer';
|
|||||||
import { ToastService } from 'src/app/services/toast.service';
|
import { ToastService } from 'src/app/services/toast.service';
|
||||||
import { TextComponent } from '../common/input/text/text.component';
|
import { TextComponent } from '../common/input/text/text.component';
|
||||||
import { SettingsService, SETTINGS_KEYS } from 'src/app/services/settings.service';
|
import { SettingsService, SETTINGS_KEYS } from 'src/app/services/settings.service';
|
||||||
|
import { PaperlessDocumentSuggestions } from 'src/app/data/paperless-document-suggestions';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-document-detail',
|
selector: 'app-document-detail',
|
||||||
@ -40,6 +41,8 @@ export class DocumentDetailComponent implements OnInit {
|
|||||||
documentId: number
|
documentId: number
|
||||||
document: PaperlessDocument
|
document: PaperlessDocument
|
||||||
metadata: PaperlessDocumentMetadata
|
metadata: PaperlessDocumentMetadata
|
||||||
|
suggestions: PaperlessDocumentSuggestions
|
||||||
|
|
||||||
title: string
|
title: string
|
||||||
previewUrl: string
|
previewUrl: string
|
||||||
downloadUrl: string
|
downloadUrl: string
|
||||||
@ -95,6 +98,7 @@ export class DocumentDetailComponent implements OnInit {
|
|||||||
this.previewUrl = this.documentsService.getPreviewUrl(this.documentId)
|
this.previewUrl = this.documentsService.getPreviewUrl(this.documentId)
|
||||||
this.downloadUrl = this.documentsService.getDownloadUrl(this.documentId)
|
this.downloadUrl = this.documentsService.getDownloadUrl(this.documentId)
|
||||||
this.downloadOriginalUrl = this.documentsService.getDownloadUrl(this.documentId, true)
|
this.downloadOriginalUrl = this.documentsService.getDownloadUrl(this.documentId, true)
|
||||||
|
this.suggestions = null
|
||||||
if (this.openDocumentService.getOpenDocument(this.documentId)) {
|
if (this.openDocumentService.getOpenDocument(this.documentId)) {
|
||||||
this.updateComponent(this.openDocumentService.getOpenDocument(this.documentId))
|
this.updateComponent(this.openDocumentService.getOpenDocument(this.documentId))
|
||||||
} else {
|
} else {
|
||||||
@ -112,6 +116,9 @@ export class DocumentDetailComponent implements OnInit {
|
|||||||
this.documentsService.getMetadata(doc.id).subscribe(result => {
|
this.documentsService.getMetadata(doc.id).subscribe(result => {
|
||||||
this.metadata = result
|
this.metadata = result
|
||||||
})
|
})
|
||||||
|
this.documentsService.getSuggestions(doc.id).subscribe(result => {
|
||||||
|
this.suggestions = result
|
||||||
|
})
|
||||||
this.title = this.documentTitlePipe.transform(doc.title)
|
this.title = this.documentTitlePipe.transform(doc.title)
|
||||||
this.documentForm.patchValue(doc)
|
this.documentForm.patchValue(doc)
|
||||||
}
|
}
|
||||||
|
@ -109,7 +109,7 @@ export class BulkEditorComponent {
|
|||||||
if (items.length == 0) {
|
if (items.length == 0) {
|
||||||
return ""
|
return ""
|
||||||
} else if (items.length == 1) {
|
} else if (items.length == 1) {
|
||||||
return items[0].name
|
return $localize`"${items[0].name}"`
|
||||||
} else if (items.length == 2) {
|
} else if (items.length == 2) {
|
||||||
return $localize`:This is for messages like 'modify "tag1" and "tag2"':"${items[0].name}" and "${items[1].name}"`
|
return $localize`:This is for messages like 'modify "tag1" and "tag2"':"${items[0].name}" and "${items[1].name}"`
|
||||||
} else {
|
} else {
|
||||||
|
9
src-ui/src/app/data/paperless-document-suggestions.ts
Normal file
9
src-ui/src/app/data/paperless-document-suggestions.ts
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
export interface PaperlessDocumentSuggestions {
|
||||||
|
|
||||||
|
tags?: number[]
|
||||||
|
|
||||||
|
correspondents?: number[]
|
||||||
|
|
||||||
|
document_types?: number[]
|
||||||
|
|
||||||
|
}
|
@ -11,6 +11,7 @@ import { CorrespondentService } from './correspondent.service';
|
|||||||
import { DocumentTypeService } from './document-type.service';
|
import { DocumentTypeService } from './document-type.service';
|
||||||
import { TagService } from './tag.service';
|
import { TagService } from './tag.service';
|
||||||
import { FILTER_RULE_TYPES } from 'src/app/data/filter-rule-type';
|
import { FILTER_RULE_TYPES } from 'src/app/data/filter-rule-type';
|
||||||
|
import { PaperlessDocumentSuggestions } from 'src/app/data/paperless-document-suggestions';
|
||||||
|
|
||||||
export const DOCUMENT_SORT_FIELDS = [
|
export const DOCUMENT_SORT_FIELDS = [
|
||||||
{ field: 'archive_serial_number', name: $localize`ASN` },
|
{ field: 'archive_serial_number', name: $localize`ASN` },
|
||||||
@ -129,4 +130,8 @@ export class DocumentService extends AbstractPaperlessService<PaperlessDocument>
|
|||||||
return this.http.post<SelectionData>(this.getResourceUrl(null, 'selection_data'), {"documents": ids})
|
return this.http.post<SelectionData>(this.getResourceUrl(null, 'selection_data'), {"documents": ids})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getSuggestions(id: number): Observable<PaperlessDocumentSuggestions> {
|
||||||
|
return this.http.get<PaperlessDocumentSuggestions>(this.getResourceUrl(id, 'suggestions'))
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -147,7 +147,7 @@
|
|||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="1b051734b0ee9021991c91b3ed4e81c244322462">
|
<trans-unit datatype="html" id="1b051734b0ee9021991c91b3ed4e81c244322462">
|
||||||
<source>Created</source>
|
<source>Created</source>
|
||||||
<target>Erstellt</target>
|
<target>Ausgestellt</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/document-list/document-list.component.html</context>
|
<context context-type="sourcefile">src/app/components/document-list/document-list.component.html</context>
|
||||||
<context context-type="linenumber">129</context>
|
<context context-type="linenumber">129</context>
|
||||||
@ -166,7 +166,7 @@
|
|||||||
<target>Löschen bestätigen</target>
|
<target>Löschen bestätigen</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
||||||
<context context-type="linenumber">192</context>
|
<context context-type="linenumber">199</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="5382975254277698192">
|
<trans-unit datatype="html" id="5382975254277698192">
|
||||||
@ -174,7 +174,7 @@
|
|||||||
<target>Möchten Sie das Dokument "<x equiv-text="this.document.title" id="PH"/>" wirklich löschen?</target>
|
<target>Möchten Sie das Dokument "<x equiv-text="this.document.title" id="PH"/>" wirklich löschen?</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
||||||
<context context-type="linenumber">193</context>
|
<context context-type="linenumber">200</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="6691075929777935948">
|
<trans-unit datatype="html" id="6691075929777935948">
|
||||||
@ -182,7 +182,7 @@
|
|||||||
<target>Die Dateien dieses Dokuments werden permanent gelöscht. Diese Aktion kann nicht rückgängig gemacht werden.</target>
|
<target>Die Dateien dieses Dokuments werden permanent gelöscht. Diese Aktion kann nicht rückgängig gemacht werden.</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
||||||
<context context-type="linenumber">194</context>
|
<context context-type="linenumber">201</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="719892092227206532">
|
<trans-unit datatype="html" id="719892092227206532">
|
||||||
@ -190,7 +190,7 @@
|
|||||||
<target>Dokument löschen</target>
|
<target>Dokument löschen</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
||||||
<context context-type="linenumber">196</context>
|
<context context-type="linenumber">203</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="1844801255494293730">
|
<trans-unit datatype="html" id="1844801255494293730">
|
||||||
@ -198,7 +198,7 @@
|
|||||||
<target>Fehler beim Löschen des Dokuments: <x equiv-text="JSON.stringify(error)" id="PH"/></target>
|
<target>Fehler beim Löschen des Dokuments: <x equiv-text="JSON.stringify(error)" id="PH"/></target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
||||||
<context context-type="linenumber">203</context>
|
<context context-type="linenumber">210</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="826b25211922a1b46436589233cb6f1a163d89b7">
|
<trans-unit datatype="html" id="826b25211922a1b46436589233cb6f1a163d89b7">
|
||||||
@ -307,7 +307,7 @@
|
|||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="30eebc2dd656dbcb259d8d5286d244ee397d63bd">
|
<trans-unit datatype="html" id="30eebc2dd656dbcb259d8d5286d244ee397d63bd">
|
||||||
<source>Date created</source>
|
<source>Date created</source>
|
||||||
<target>Erstellt am</target>
|
<target>Ausgestellt am</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context>
|
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.html</context>
|
||||||
<context context-type="linenumber">61</context>
|
<context context-type="linenumber">61</context>
|
||||||
@ -1283,6 +1283,14 @@
|
|||||||
<context context-type="linenumber">73</context>
|
<context context-type="linenumber">73</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
|
<trans-unit datatype="html" id="7894972847287473517">
|
||||||
|
<source>"<x equiv-text="items[0].name" id="PH"/>"</source>
|
||||||
|
<target>"<x equiv-text="items[0].name" id="PH"/>"</target>
|
||||||
|
<context-group purpose="location">
|
||||||
|
<context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.ts</context>
|
||||||
|
<context context-type="linenumber">112</context>
|
||||||
|
</context-group>
|
||||||
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="8639884465898458690">
|
<trans-unit datatype="html" id="8639884465898458690">
|
||||||
<source>"<x equiv-text="items[0].name" id="PH"/>" and "<x equiv-text="items[1].name" id="PH_1"/>"</source>
|
<source>"<x equiv-text="items[0].name" id="PH"/>" and "<x equiv-text="items[1].name" id="PH_1"/>"</source>
|
||||||
<target>"<x equiv-text="items[0].name" id="PH"/>" und "<x equiv-text="items[1].name" id="PH_1"/>"</target>
|
<target>"<x equiv-text="items[0].name" id="PH"/>" und "<x equiv-text="items[1].name" id="PH_1"/>"</target>
|
||||||
@ -1292,14 +1300,6 @@
|
|||||||
</context-group>
|
</context-group>
|
||||||
<note from="description" priority="1">This is for messages like 'modify "tag1" and "tag2"'</note>
|
<note from="description" priority="1">This is for messages like 'modify "tag1" and "tag2"'</note>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="7894972847287473517">
|
|
||||||
<source>"<x equiv-text="i.name" id="PH"/>"</source>
|
|
||||||
<target>"<x equiv-text="i.name" id="PH"/>"</target>
|
|
||||||
<context-group purpose="location">
|
|
||||||
<context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.ts</context>
|
|
||||||
<context context-type="linenumber">116</context>
|
|
||||||
</context-group>
|
|
||||||
</trans-unit>
|
|
||||||
<trans-unit datatype="html" id="760986369763309193">
|
<trans-unit datatype="html" id="760986369763309193">
|
||||||
<source>, </source>
|
<source>, </source>
|
||||||
<target>, </target>
|
<target>, </target>
|
||||||
@ -1470,6 +1470,14 @@
|
|||||||
<context context-type="linenumber">27</context>
|
<context context-type="linenumber">27</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
|
<trans-unit datatype="html" id="a1e6c11f20d4bf6e8e6b43e3c6d2561b2080645e">
|
||||||
|
<source>Suggestions:</source>
|
||||||
|
<target>Vorschläge:</target>
|
||||||
|
<context-group purpose="location">
|
||||||
|
<context context-type="sourcefile">src/app/components/common/input/select/select.component.html</context>
|
||||||
|
<context context-type="linenumber">26</context>
|
||||||
|
</context-group>
|
||||||
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="27d158b47717ff9305d19866960418c603f19d55">
|
<trans-unit datatype="html" id="27d158b47717ff9305d19866960418c603f19d55">
|
||||||
<source>Save current view</source>
|
<source>Save current view</source>
|
||||||
<target>Aktuelle Ansicht speichern</target>
|
<target>Aktuelle Ansicht speichern</target>
|
||||||
@ -1723,7 +1731,7 @@
|
|||||||
<target>ASN</target>
|
<target>ASN</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">16</context>
|
<context context-type="linenumber">17</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="2691296884221415710">
|
<trans-unit datatype="html" id="2691296884221415710">
|
||||||
@ -1731,7 +1739,7 @@
|
|||||||
<target>Korrespondent</target>
|
<target>Korrespondent</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">17</context>
|
<context context-type="linenumber">18</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="5701618810648052610">
|
<trans-unit datatype="html" id="5701618810648052610">
|
||||||
@ -1739,7 +1747,7 @@
|
|||||||
<target>Titel</target>
|
<target>Titel</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">18</context>
|
<context context-type="linenumber">19</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="5066119607229701477">
|
<trans-unit datatype="html" id="5066119607229701477">
|
||||||
@ -1747,15 +1755,15 @@
|
|||||||
<target>Dokumenttyp</target>
|
<target>Dokumenttyp</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">19</context>
|
<context context-type="linenumber">20</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="4207916966377787111">
|
<trans-unit datatype="html" id="4207916966377787111">
|
||||||
<source>Created</source>
|
<source>Created</source>
|
||||||
<target>Erstellt am</target>
|
<target>Ausgestellt am</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">20</context>
|
<context context-type="linenumber">21</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="231679111972850796">
|
<trans-unit datatype="html" id="231679111972850796">
|
||||||
@ -1763,7 +1771,7 @@
|
|||||||
<target>Hinzugefügt am</target>
|
<target>Hinzugefügt am</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">21</context>
|
<context context-type="linenumber">22</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="3553216189604488439">
|
<trans-unit datatype="html" id="3553216189604488439">
|
||||||
@ -1771,7 +1779,7 @@
|
|||||||
<target>Geändert am</target>
|
<target>Geändert am</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">22</context>
|
<context context-type="linenumber">23</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="2056433880533904076">
|
<trans-unit datatype="html" id="2056433880533904076">
|
||||||
|
@ -166,7 +166,7 @@
|
|||||||
<target>Confirmer la suppression</target>
|
<target>Confirmer la suppression</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
||||||
<context context-type="linenumber">192</context>
|
<context context-type="linenumber">199</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="5382975254277698192">
|
<trans-unit datatype="html" id="5382975254277698192">
|
||||||
@ -174,7 +174,7 @@
|
|||||||
<target>Voulez-vous vraiment supprimer le document "<x equiv-text="this.document.title" id="PH"/>" ?</target>
|
<target>Voulez-vous vraiment supprimer le document "<x equiv-text="this.document.title" id="PH"/>" ?</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
||||||
<context context-type="linenumber">193</context>
|
<context context-type="linenumber">200</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="6691075929777935948">
|
<trans-unit datatype="html" id="6691075929777935948">
|
||||||
@ -182,7 +182,7 @@
|
|||||||
<target>Les fichiers liés à ce document seront supprimés définitivement. Cette action est irréversible.</target>
|
<target>Les fichiers liés à ce document seront supprimés définitivement. Cette action est irréversible.</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
||||||
<context context-type="linenumber">194</context>
|
<context context-type="linenumber">201</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="719892092227206532">
|
<trans-unit datatype="html" id="719892092227206532">
|
||||||
@ -190,7 +190,7 @@
|
|||||||
<target>Supprimer le document</target>
|
<target>Supprimer le document</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
||||||
<context context-type="linenumber">196</context>
|
<context context-type="linenumber">203</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="1844801255494293730">
|
<trans-unit datatype="html" id="1844801255494293730">
|
||||||
@ -198,7 +198,7 @@
|
|||||||
<target>Une erreur s'est produite lors de la suppression du document : <x equiv-text="JSON.stringify(error)" id="PH"/></target>
|
<target>Une erreur s'est produite lors de la suppression du document : <x equiv-text="JSON.stringify(error)" id="PH"/></target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
<context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context>
|
||||||
<context context-type="linenumber">203</context>
|
<context context-type="linenumber">210</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="826b25211922a1b46436589233cb6f1a163d89b7">
|
<trans-unit datatype="html" id="826b25211922a1b46436589233cb6f1a163d89b7">
|
||||||
@ -1283,6 +1283,14 @@
|
|||||||
<context context-type="linenumber">73</context>
|
<context context-type="linenumber">73</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
|
<trans-unit datatype="html" id="7894972847287473517">
|
||||||
|
<source>"<x equiv-text="items[0].name" id="PH"/>"</source>
|
||||||
|
<target>"<x equiv-text="items[0].name" id="PH"/>"</target>
|
||||||
|
<context-group purpose="location">
|
||||||
|
<context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.ts</context>
|
||||||
|
<context context-type="linenumber">112</context>
|
||||||
|
</context-group>
|
||||||
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="8639884465898458690">
|
<trans-unit datatype="html" id="8639884465898458690">
|
||||||
<source>"<x equiv-text="items[0].name" id="PH"/>" and "<x equiv-text="items[1].name" id="PH_1"/>"</source>
|
<source>"<x equiv-text="items[0].name" id="PH"/>" and "<x equiv-text="items[1].name" id="PH_1"/>"</source>
|
||||||
<target>"<x equiv-text="items[0].name" id="PH"/>" et "<x equiv-text="items[1].name" id="PH_1"/>"</target>
|
<target>"<x equiv-text="items[0].name" id="PH"/>" et "<x equiv-text="items[1].name" id="PH_1"/>"</target>
|
||||||
@ -1292,14 +1300,6 @@
|
|||||||
</context-group>
|
</context-group>
|
||||||
<note from="description" priority="1">This is for messages like 'modify "tag1" and "tag2"'</note>
|
<note from="description" priority="1">This is for messages like 'modify "tag1" and "tag2"'</note>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="7894972847287473517">
|
|
||||||
<source>"<x equiv-text="i.name" id="PH"/>"</source>
|
|
||||||
<target>"<x equiv-text="i.name" id="PH"/>"</target>
|
|
||||||
<context-group purpose="location">
|
|
||||||
<context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.ts</context>
|
|
||||||
<context context-type="linenumber">116</context>
|
|
||||||
</context-group>
|
|
||||||
</trans-unit>
|
|
||||||
<trans-unit datatype="html" id="760986369763309193">
|
<trans-unit datatype="html" id="760986369763309193">
|
||||||
<source>, </source>
|
<source>, </source>
|
||||||
<target>, </target>
|
<target>, </target>
|
||||||
@ -1470,6 +1470,14 @@
|
|||||||
<context context-type="linenumber">27</context>
|
<context context-type="linenumber">27</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
|
<trans-unit datatype="html" id="a1e6c11f20d4bf6e8e6b43e3c6d2561b2080645e">
|
||||||
|
<source>Suggestions:</source>
|
||||||
|
<target>Suggestions : </target>
|
||||||
|
<context-group purpose="location">
|
||||||
|
<context context-type="sourcefile">src/app/components/common/input/select/select.component.html</context>
|
||||||
|
<context context-type="linenumber">26</context>
|
||||||
|
</context-group>
|
||||||
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="27d158b47717ff9305d19866960418c603f19d55">
|
<trans-unit datatype="html" id="27d158b47717ff9305d19866960418c603f19d55">
|
||||||
<source>Save current view</source>
|
<source>Save current view</source>
|
||||||
<target>Enregistrer la vue actuelle</target>
|
<target>Enregistrer la vue actuelle</target>
|
||||||
@ -1723,7 +1731,7 @@
|
|||||||
<target>NSA</target>
|
<target>NSA</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">16</context>
|
<context context-type="linenumber">17</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="2691296884221415710">
|
<trans-unit datatype="html" id="2691296884221415710">
|
||||||
@ -1731,7 +1739,7 @@
|
|||||||
<target>Correspondant</target>
|
<target>Correspondant</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">17</context>
|
<context context-type="linenumber">18</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="5701618810648052610">
|
<trans-unit datatype="html" id="5701618810648052610">
|
||||||
@ -1739,7 +1747,7 @@
|
|||||||
<target>Titre</target>
|
<target>Titre</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">18</context>
|
<context context-type="linenumber">19</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="5066119607229701477">
|
<trans-unit datatype="html" id="5066119607229701477">
|
||||||
@ -1747,7 +1755,7 @@
|
|||||||
<target>Type de document</target>
|
<target>Type de document</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">19</context>
|
<context context-type="linenumber">20</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="4207916966377787111">
|
<trans-unit datatype="html" id="4207916966377787111">
|
||||||
@ -1755,7 +1763,7 @@
|
|||||||
<target>Date de création</target>
|
<target>Date de création</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">20</context>
|
<context context-type="linenumber">21</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="231679111972850796">
|
<trans-unit datatype="html" id="231679111972850796">
|
||||||
@ -1763,7 +1771,7 @@
|
|||||||
<target>Date d'ajout</target>
|
<target>Date d'ajout</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">21</context>
|
<context context-type="linenumber">22</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="3553216189604488439">
|
<trans-unit datatype="html" id="3553216189604488439">
|
||||||
@ -1771,7 +1779,7 @@
|
|||||||
<target>Date de modification</target>
|
<target>Date de modification</target>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
<context context-type="sourcefile">src/app/services/rest/document.service.ts</context>
|
||||||
<context context-type="linenumber">22</context>
|
<context context-type="linenumber">23</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit datatype="html" id="2056433880533904076">
|
<trans-unit datatype="html" id="2056433880533904076">
|
||||||
|
@ -26,6 +26,34 @@ def preprocess_content(content):
|
|||||||
return content
|
return content
|
||||||
|
|
||||||
|
|
||||||
|
def load_classifier():
|
||||||
|
if not os.path.isfile(settings.MODEL_FILE):
|
||||||
|
logger.debug(
|
||||||
|
f"Document classification model does not exist (yet), not "
|
||||||
|
f"performing automatic matching."
|
||||||
|
)
|
||||||
|
return None
|
||||||
|
|
||||||
|
try:
|
||||||
|
classifier = DocumentClassifier()
|
||||||
|
classifier.reload()
|
||||||
|
except (EOFError, IncompatibleClassifierVersionError) as e:
|
||||||
|
# there's something wrong with the model file.
|
||||||
|
logger.error(
|
||||||
|
f"Unrecoverable error while loading document "
|
||||||
|
f"classification model: {str(e)}, deleting model file."
|
||||||
|
)
|
||||||
|
os.unlink(settings.MODEL_FILE)
|
||||||
|
classifier = None
|
||||||
|
except OSError as e:
|
||||||
|
logger.error(
|
||||||
|
f"Error while loading document classification model: {str(e)}"
|
||||||
|
)
|
||||||
|
classifier = None
|
||||||
|
|
||||||
|
return classifier
|
||||||
|
|
||||||
|
|
||||||
class DocumentClassifier(object):
|
class DocumentClassifier(object):
|
||||||
|
|
||||||
FORMAT_VERSION = 6
|
FORMAT_VERSION = 6
|
||||||
|
@ -14,7 +14,7 @@ from django.utils import timezone
|
|||||||
from filelock import FileLock
|
from filelock import FileLock
|
||||||
from rest_framework.reverse import reverse
|
from rest_framework.reverse import reverse
|
||||||
|
|
||||||
from .classifier import DocumentClassifier, IncompatibleClassifierVersionError
|
from .classifier import load_classifier
|
||||||
from .file_handling import create_source_path_directory, \
|
from .file_handling import create_source_path_directory, \
|
||||||
generate_unique_filename
|
generate_unique_filename
|
||||||
from .loggers import LoggingMixin
|
from .loggers import LoggingMixin
|
||||||
@ -262,14 +262,8 @@ class Consumer(LoggingMixin):
|
|||||||
# reloading the classifier multiple times, since there are multiple
|
# reloading the classifier multiple times, since there are multiple
|
||||||
# post-consume hooks that all require the classifier.
|
# post-consume hooks that all require the classifier.
|
||||||
|
|
||||||
try:
|
classifier = load_classifier()
|
||||||
classifier = DocumentClassifier()
|
|
||||||
classifier.reload()
|
|
||||||
except (OSError, EOFError, IncompatibleClassifierVersionError) as e:
|
|
||||||
self.log(
|
|
||||||
"warning",
|
|
||||||
f"Cannot classify documents: {e}.")
|
|
||||||
classifier = None
|
|
||||||
self._send_progress(95, 100, 'WORKING', MESSAGE_SAVE_DOCUMENT)
|
self._send_progress(95, 100, 'WORKING', MESSAGE_SAVE_DOCUMENT)
|
||||||
# now that everything is done, we can start to store the document
|
# now that everything is done, we can start to store the document
|
||||||
# in the system. This will be a transaction and reasonably fast.
|
# in the system. This will be a transaction and reasonably fast.
|
||||||
|
@ -2,8 +2,7 @@ import logging
|
|||||||
|
|
||||||
from django.core.management.base import BaseCommand
|
from django.core.management.base import BaseCommand
|
||||||
|
|
||||||
from documents.classifier import DocumentClassifier, \
|
from documents.classifier import load_classifier
|
||||||
IncompatibleClassifierVersionError
|
|
||||||
from documents.models import Document
|
from documents.models import Document
|
||||||
from ...mixins import Renderable
|
from ...mixins import Renderable
|
||||||
from ...signals.handlers import set_correspondent, set_document_type, set_tags
|
from ...signals.handlers import set_correspondent, set_document_type, set_tags
|
||||||
@ -70,13 +69,7 @@ class Command(Renderable, BaseCommand):
|
|||||||
queryset = Document.objects.all()
|
queryset = Document.objects.all()
|
||||||
documents = queryset.distinct()
|
documents = queryset.distinct()
|
||||||
|
|
||||||
classifier = DocumentClassifier()
|
classifier = load_classifier()
|
||||||
try:
|
|
||||||
classifier.reload()
|
|
||||||
except (OSError, EOFError, IncompatibleClassifierVersionError) as e:
|
|
||||||
logging.getLogger(__name__).warning(
|
|
||||||
f"Cannot classify documents: {e}.")
|
|
||||||
classifier = None
|
|
||||||
|
|
||||||
for document in documents:
|
for document in documents:
|
||||||
logging.getLogger(__name__).info(
|
logging.getLogger(__name__).info(
|
||||||
|
@ -6,10 +6,9 @@ from django.db.models.signals import post_save
|
|||||||
from whoosh.writing import AsyncWriter
|
from whoosh.writing import AsyncWriter
|
||||||
|
|
||||||
from documents import index, sanity_checker
|
from documents import index, sanity_checker
|
||||||
from documents.classifier import DocumentClassifier, \
|
from documents.classifier import DocumentClassifier, load_classifier
|
||||||
IncompatibleClassifierVersionError
|
|
||||||
from documents.consumer import Consumer, ConsumerError
|
from documents.consumer import Consumer, ConsumerError
|
||||||
from documents.models import Document
|
from documents.models import Document, Tag, DocumentType, Correspondent
|
||||||
from documents.sanity_checker import SanityFailedError
|
from documents.sanity_checker import SanityFailedError
|
||||||
|
|
||||||
|
|
||||||
@ -30,13 +29,18 @@ def index_reindex():
|
|||||||
|
|
||||||
|
|
||||||
def train_classifier():
|
def train_classifier():
|
||||||
classifier = DocumentClassifier()
|
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()):
|
||||||
|
|
||||||
try:
|
return
|
||||||
# load the classifier, since we might not have to train it again.
|
|
||||||
classifier.reload()
|
classifier = load_classifier()
|
||||||
except (OSError, EOFError, IncompatibleClassifierVersionError):
|
|
||||||
# This is what we're going to fix here.
|
if not classifier:
|
||||||
classifier = DocumentClassifier()
|
classifier = DocumentClassifier()
|
||||||
|
|
||||||
try:
|
try:
|
||||||
@ -52,7 +56,7 @@ def train_classifier():
|
|||||||
)
|
)
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logging.getLogger(__name__).error(
|
logging.getLogger(__name__).warning(
|
||||||
"Classifier error: " + str(e)
|
"Classifier error: " + str(e)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -590,6 +590,10 @@ class TestDocumentApi(DirectoriesMixin, APITestCase):
|
|||||||
self.assertEqual(len(meta['original_metadata']), 0)
|
self.assertEqual(len(meta['original_metadata']), 0)
|
||||||
self.assertGreater(len(meta['archive_metadata']), 0)
|
self.assertGreater(len(meta['archive_metadata']), 0)
|
||||||
|
|
||||||
|
def test_get_metadata_invalid_doc(self):
|
||||||
|
response = self.client.get(f"/api/documents/34576/metadata/")
|
||||||
|
self.assertEqual(response.status_code, 404)
|
||||||
|
|
||||||
def test_get_metadata_no_archive(self):
|
def test_get_metadata_no_archive(self):
|
||||||
doc = Document.objects.create(title="test", filename="file.pdf", mime_type="application/pdf")
|
doc = Document.objects.create(title="test", filename="file.pdf", mime_type="application/pdf")
|
||||||
|
|
||||||
@ -605,6 +609,30 @@ class TestDocumentApi(DirectoriesMixin, APITestCase):
|
|||||||
self.assertGreater(len(meta['original_metadata']), 0)
|
self.assertGreater(len(meta['original_metadata']), 0)
|
||||||
self.assertIsNone(meta['archive_metadata'])
|
self.assertIsNone(meta['archive_metadata'])
|
||||||
|
|
||||||
|
def test_get_empty_suggestions(self):
|
||||||
|
doc = Document.objects.create(title="test", mime_type="application/pdf")
|
||||||
|
|
||||||
|
response = self.client.get(f"/api/documents/{doc.pk}/suggestions/")
|
||||||
|
|
||||||
|
self.assertEqual(response.status_code, 200)
|
||||||
|
self.assertEqual(response.data, {'correspondents': [], 'tags': [], 'document_types': []})
|
||||||
|
|
||||||
|
def test_get_suggestions_invalid_doc(self):
|
||||||
|
response = self.client.get(f"/api/documents/34676/suggestions/")
|
||||||
|
self.assertEqual(response.status_code, 404)
|
||||||
|
|
||||||
|
@mock.patch("documents.views.match_correspondents")
|
||||||
|
@mock.patch("documents.views.match_tags")
|
||||||
|
@mock.patch("documents.views.match_document_types")
|
||||||
|
def test_get_suggestions(self, match_document_types, match_tags, match_correspondents):
|
||||||
|
doc = Document.objects.create(title="test", mime_type="application/pdf", content="this is an invoice!")
|
||||||
|
match_tags.return_value = [Tag(id=56), Tag(id=123)]
|
||||||
|
match_document_types.return_value = [DocumentType(id=23)]
|
||||||
|
match_correspondents.return_value = [Correspondent(id=88), Correspondent(id=2)]
|
||||||
|
|
||||||
|
response = self.client.get(f"/api/documents/{doc.pk}/suggestions/")
|
||||||
|
self.assertEqual(response.data, {'correspondents': [88,2], 'tags': [56,123], 'document_types': [23]})
|
||||||
|
|
||||||
def test_saved_views(self):
|
def test_saved_views(self):
|
||||||
u1 = User.objects.create_user("user1")
|
u1 = User.objects.create_user("user1")
|
||||||
u2 = User.objects.create_user("user2")
|
u2 = User.objects.create_user("user2")
|
||||||
|
@ -1,10 +1,13 @@
|
|||||||
|
import os
|
||||||
import tempfile
|
import tempfile
|
||||||
|
from pathlib import Path
|
||||||
from time import sleep
|
from time import sleep
|
||||||
from unittest import mock
|
from unittest import mock
|
||||||
|
|
||||||
|
from django.conf import settings
|
||||||
from django.test import TestCase, override_settings
|
from django.test import TestCase, override_settings
|
||||||
|
|
||||||
from documents.classifier import DocumentClassifier, IncompatibleClassifierVersionError
|
from documents.classifier import DocumentClassifier, IncompatibleClassifierVersionError, load_classifier
|
||||||
from documents.models import Correspondent, Document, Tag, DocumentType
|
from documents.models import Correspondent, Document, Tag, DocumentType
|
||||||
from documents.tests.utils import DirectoriesMixin
|
from documents.tests.utils import DirectoriesMixin
|
||||||
|
|
||||||
@ -235,3 +238,30 @@ class TestClassifier(DirectoriesMixin, TestCase):
|
|||||||
self.classifier.train()
|
self.classifier.train()
|
||||||
self.assertListEqual(self.classifier.predict_tags(doc1.content), [t1.pk])
|
self.assertListEqual(self.classifier.predict_tags(doc1.content), [t1.pk])
|
||||||
self.assertListEqual(self.classifier.predict_tags(doc2.content), [])
|
self.assertListEqual(self.classifier.predict_tags(doc2.content), [])
|
||||||
|
|
||||||
|
def test_load_classifier_not_exists(self):
|
||||||
|
self.assertFalse(os.path.exists(settings.MODEL_FILE))
|
||||||
|
self.assertIsNone(load_classifier())
|
||||||
|
|
||||||
|
@mock.patch("documents.classifier.DocumentClassifier.reload")
|
||||||
|
def test_load_classifier(self, reload):
|
||||||
|
Path(settings.MODEL_FILE).touch()
|
||||||
|
self.assertIsNotNone(load_classifier())
|
||||||
|
|
||||||
|
@mock.patch("documents.classifier.DocumentClassifier.reload")
|
||||||
|
def test_load_classifier_incompatible_version(self, reload):
|
||||||
|
Path(settings.MODEL_FILE).touch()
|
||||||
|
self.assertTrue(os.path.exists(settings.MODEL_FILE))
|
||||||
|
|
||||||
|
reload.side_effect = IncompatibleClassifierVersionError()
|
||||||
|
self.assertIsNone(load_classifier())
|
||||||
|
self.assertFalse(os.path.exists(settings.MODEL_FILE))
|
||||||
|
|
||||||
|
@mock.patch("documents.classifier.DocumentClassifier.reload")
|
||||||
|
def test_load_classifier_os_error(self, reload):
|
||||||
|
Path(settings.MODEL_FILE).touch()
|
||||||
|
self.assertTrue(os.path.exists(settings.MODEL_FILE))
|
||||||
|
|
||||||
|
reload.side_effect = OSError()
|
||||||
|
self.assertIsNone(load_classifier())
|
||||||
|
self.assertTrue(os.path.exists(settings.MODEL_FILE))
|
||||||
|
@ -460,7 +460,7 @@ class TestConsumer(DirectoriesMixin, TestCase):
|
|||||||
|
|
||||||
self._assert_first_last_send_progress()
|
self._assert_first_last_send_progress()
|
||||||
|
|
||||||
@mock.patch("documents.consumer.DocumentClassifier")
|
@mock.patch("documents.consumer.load_classifier")
|
||||||
def testClassifyDocument(self, m):
|
def testClassifyDocument(self, m):
|
||||||
correspondent = Correspondent.objects.create(name="test")
|
correspondent = Correspondent.objects.create(name="test")
|
||||||
dtype = DocumentType.objects.create(name="test")
|
dtype = DocumentType.objects.create(name="test")
|
||||||
|
@ -20,7 +20,7 @@ class TestSettings(TestCase):
|
|||||||
self.assertEqual(default_threads, 1)
|
self.assertEqual(default_threads, 1)
|
||||||
|
|
||||||
def test_workers_threads(self):
|
def test_workers_threads(self):
|
||||||
for i in range(2, 64):
|
for i in range(1, 64):
|
||||||
with mock.patch("paperless.settings.multiprocessing.cpu_count") as cpu_count:
|
with mock.patch("paperless.settings.multiprocessing.cpu_count") as cpu_count:
|
||||||
cpu_count.return_value = i
|
cpu_count.return_value = i
|
||||||
|
|
||||||
@ -31,4 +31,4 @@ class TestSettings(TestCase):
|
|||||||
self.assertTrue(default_workers >= 1)
|
self.assertTrue(default_workers >= 1)
|
||||||
self.assertTrue(default_threads >= 1)
|
self.assertTrue(default_threads >= 1)
|
||||||
|
|
||||||
self.assertTrue(default_workers * default_threads < i, f"{i}")
|
self.assertTrue(default_workers * default_threads <= i, f"{i}")
|
||||||
|
@ -1,11 +1,12 @@
|
|||||||
from datetime import datetime
|
import os
|
||||||
from unittest import mock
|
from unittest import mock
|
||||||
|
|
||||||
|
from django.conf import settings
|
||||||
from django.test import TestCase
|
from django.test import TestCase
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
|
|
||||||
from documents import tasks
|
from documents import tasks
|
||||||
from documents.models import Document
|
from documents.models import Document, Tag, Correspondent, DocumentType
|
||||||
from documents.sanity_checker import SanityError, SanityFailedError
|
from documents.sanity_checker import SanityError, SanityFailedError
|
||||||
from documents.tests.utils import DirectoriesMixin
|
from documents.tests.utils import DirectoriesMixin
|
||||||
|
|
||||||
@ -22,8 +23,55 @@ class TestTasks(DirectoriesMixin, TestCase):
|
|||||||
|
|
||||||
tasks.index_optimize()
|
tasks.index_optimize()
|
||||||
|
|
||||||
def test_train_classifier(self):
|
@mock.patch("documents.tasks.load_classifier")
|
||||||
|
def test_train_classifier_no_auto_matching(self, load_classifier):
|
||||||
tasks.train_classifier()
|
tasks.train_classifier()
|
||||||
|
load_classifier.assert_not_called()
|
||||||
|
|
||||||
|
@mock.patch("documents.tasks.load_classifier")
|
||||||
|
def test_train_classifier_with_auto_tag(self, load_classifier):
|
||||||
|
load_classifier.return_value = None
|
||||||
|
Tag.objects.create(matching_algorithm=Tag.MATCH_AUTO, name="test")
|
||||||
|
tasks.train_classifier()
|
||||||
|
load_classifier.assert_called_once()
|
||||||
|
self.assertFalse(os.path.isfile(settings.MODEL_FILE))
|
||||||
|
|
||||||
|
@mock.patch("documents.tasks.load_classifier")
|
||||||
|
def test_train_classifier_with_auto_type(self, load_classifier):
|
||||||
|
load_classifier.return_value = None
|
||||||
|
DocumentType.objects.create(matching_algorithm=Tag.MATCH_AUTO, name="test")
|
||||||
|
tasks.train_classifier()
|
||||||
|
load_classifier.assert_called_once()
|
||||||
|
self.assertFalse(os.path.isfile(settings.MODEL_FILE))
|
||||||
|
|
||||||
|
@mock.patch("documents.tasks.load_classifier")
|
||||||
|
def test_train_classifier_with_auto_correspondent(self, load_classifier):
|
||||||
|
load_classifier.return_value = None
|
||||||
|
Correspondent.objects.create(matching_algorithm=Tag.MATCH_AUTO, name="test")
|
||||||
|
tasks.train_classifier()
|
||||||
|
load_classifier.assert_called_once()
|
||||||
|
self.assertFalse(os.path.isfile(settings.MODEL_FILE))
|
||||||
|
|
||||||
|
def test_train_classifier(self):
|
||||||
|
c = Correspondent.objects.create(matching_algorithm=Tag.MATCH_AUTO, name="test")
|
||||||
|
doc = Document.objects.create(correspondent=c, content="test", title="test")
|
||||||
|
self.assertFalse(os.path.isfile(settings.MODEL_FILE))
|
||||||
|
|
||||||
|
tasks.train_classifier()
|
||||||
|
self.assertTrue(os.path.isfile(settings.MODEL_FILE))
|
||||||
|
mtime = os.stat(settings.MODEL_FILE).st_mtime
|
||||||
|
|
||||||
|
tasks.train_classifier()
|
||||||
|
self.assertTrue(os.path.isfile(settings.MODEL_FILE))
|
||||||
|
mtime2 = os.stat(settings.MODEL_FILE).st_mtime
|
||||||
|
self.assertEqual(mtime, mtime2)
|
||||||
|
|
||||||
|
doc.content = "test2"
|
||||||
|
doc.save()
|
||||||
|
tasks.train_classifier()
|
||||||
|
self.assertTrue(os.path.isfile(settings.MODEL_FILE))
|
||||||
|
mtime3 = os.stat(settings.MODEL_FILE).st_mtime
|
||||||
|
self.assertNotEqual(mtime2, mtime3)
|
||||||
|
|
||||||
@mock.patch("documents.tasks.sanity_checker.check_sanity")
|
@mock.patch("documents.tasks.sanity_checker.check_sanity")
|
||||||
def test_sanity_check(self, m):
|
def test_sanity_check(self, m):
|
||||||
@ -35,7 +83,7 @@ class TestTasks(DirectoriesMixin, TestCase):
|
|||||||
self.assertRaises(SanityFailedError, tasks.sanity_check)
|
self.assertRaises(SanityFailedError, tasks.sanity_check)
|
||||||
m.assert_called_once()
|
m.assert_called_once()
|
||||||
|
|
||||||
def test_culk_update_documents(self):
|
def test_bulk_update_documents(self):
|
||||||
doc1 = Document.objects.create(title="test", content="my document", checksum="wow", added=timezone.now(),
|
doc1 = Document.objects.create(title="test", content="my document", checksum="wow", added=timezone.now(),
|
||||||
created=timezone.now(), modified=timezone.now())
|
created=timezone.now(), modified=timezone.now())
|
||||||
|
|
||||||
|
@ -35,6 +35,7 @@ from rest_framework.viewsets import (
|
|||||||
import documents.index as index
|
import documents.index as index
|
||||||
from paperless.db import GnuPG
|
from paperless.db import GnuPG
|
||||||
from paperless.views import StandardPagination
|
from paperless.views import StandardPagination
|
||||||
|
from .classifier import load_classifier
|
||||||
from .filters import (
|
from .filters import (
|
||||||
CorrespondentFilterSet,
|
CorrespondentFilterSet,
|
||||||
DocumentFilterSet,
|
DocumentFilterSet,
|
||||||
@ -42,6 +43,7 @@ from .filters import (
|
|||||||
DocumentTypeFilterSet,
|
DocumentTypeFilterSet,
|
||||||
LogFilterSet
|
LogFilterSet
|
||||||
)
|
)
|
||||||
|
from .matching import match_correspondents, match_tags, match_document_types
|
||||||
from .models import Correspondent, Document, Log, Tag, DocumentType, SavedView
|
from .models import Correspondent, Document, Log, Tag, DocumentType, SavedView
|
||||||
from .parsers import get_parser_class_for_mime_type
|
from .parsers import get_parser_class_for_mime_type
|
||||||
from .serialisers import (
|
from .serialisers import (
|
||||||
@ -133,10 +135,6 @@ class DocumentTypeViewSet(ModelViewSet):
|
|||||||
ordering_fields = ("name", "matching_algorithm", "match", "document_count")
|
ordering_fields = ("name", "matching_algorithm", "match", "document_count")
|
||||||
|
|
||||||
|
|
||||||
class BulkEditForm(object):
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
class DocumentViewSet(RetrieveModelMixin,
|
class DocumentViewSet(RetrieveModelMixin,
|
||||||
UpdateModelMixin,
|
UpdateModelMixin,
|
||||||
DestroyModelMixin,
|
DestroyModelMixin,
|
||||||
@ -230,31 +228,50 @@ class DocumentViewSet(RetrieveModelMixin,
|
|||||||
def metadata(self, request, pk=None):
|
def metadata(self, request, pk=None):
|
||||||
try:
|
try:
|
||||||
doc = Document.objects.get(pk=pk)
|
doc = Document.objects.get(pk=pk)
|
||||||
|
|
||||||
meta = {
|
|
||||||
"original_checksum": doc.checksum,
|
|
||||||
"original_size": os.stat(doc.source_path).st_size,
|
|
||||||
"original_mime_type": doc.mime_type,
|
|
||||||
"media_filename": doc.filename,
|
|
||||||
"has_archive_version": os.path.isfile(doc.archive_path),
|
|
||||||
"original_metadata": self.get_metadata(
|
|
||||||
doc.source_path, doc.mime_type)
|
|
||||||
}
|
|
||||||
|
|
||||||
if doc.archive_checksum and os.path.isfile(doc.archive_path):
|
|
||||||
meta['archive_checksum'] = doc.archive_checksum
|
|
||||||
meta['archive_size'] = os.stat(doc.archive_path).st_size,
|
|
||||||
meta['archive_metadata'] = self.get_metadata(
|
|
||||||
doc.archive_path, "application/pdf")
|
|
||||||
else:
|
|
||||||
meta['archive_checksum'] = None
|
|
||||||
meta['archive_size'] = None
|
|
||||||
meta['archive_metadata'] = None
|
|
||||||
|
|
||||||
return Response(meta)
|
|
||||||
except Document.DoesNotExist:
|
except Document.DoesNotExist:
|
||||||
raise Http404()
|
raise Http404()
|
||||||
|
|
||||||
|
meta = {
|
||||||
|
"original_checksum": doc.checksum,
|
||||||
|
"original_size": os.stat(doc.source_path).st_size,
|
||||||
|
"original_mime_type": doc.mime_type,
|
||||||
|
"media_filename": doc.filename,
|
||||||
|
"has_archive_version": os.path.isfile(doc.archive_path),
|
||||||
|
"original_metadata": self.get_metadata(
|
||||||
|
doc.source_path, doc.mime_type)
|
||||||
|
}
|
||||||
|
|
||||||
|
if doc.archive_checksum and os.path.isfile(doc.archive_path):
|
||||||
|
meta['archive_checksum'] = doc.archive_checksum
|
||||||
|
meta['archive_size'] = os.stat(doc.archive_path).st_size,
|
||||||
|
meta['archive_metadata'] = self.get_metadata(
|
||||||
|
doc.archive_path, "application/pdf")
|
||||||
|
else:
|
||||||
|
meta['archive_checksum'] = None
|
||||||
|
meta['archive_size'] = None
|
||||||
|
meta['archive_metadata'] = None
|
||||||
|
|
||||||
|
return Response(meta)
|
||||||
|
|
||||||
|
@action(methods=['get'], detail=True)
|
||||||
|
def suggestions(self, request, pk=None):
|
||||||
|
try:
|
||||||
|
doc = Document.objects.get(pk=pk)
|
||||||
|
except Document.DoesNotExist:
|
||||||
|
raise Http404()
|
||||||
|
|
||||||
|
classifier = load_classifier()
|
||||||
|
|
||||||
|
return Response({
|
||||||
|
"correspondents": [
|
||||||
|
c.id for c in match_correspondents(doc, classifier)
|
||||||
|
],
|
||||||
|
"tags": [t.id for t in match_tags(doc, classifier)],
|
||||||
|
"document_types": [
|
||||||
|
dt.id for dt in match_document_types(doc, classifier)
|
||||||
|
]
|
||||||
|
})
|
||||||
|
|
||||||
@action(methods=['get'], detail=True)
|
@action(methods=['get'], detail=True)
|
||||||
def preview(self, request, pk=None):
|
def preview(self, request, pk=None):
|
||||||
try:
|
try:
|
||||||
@ -382,6 +399,7 @@ class PostDocumentView(APIView):
|
|||||||
|
|
||||||
with tempfile.NamedTemporaryFile(prefix="paperless-upload-",
|
with tempfile.NamedTemporaryFile(prefix="paperless-upload-",
|
||||||
dir=settings.SCRATCH_DIR,
|
dir=settings.SCRATCH_DIR,
|
||||||
|
buffering=0,
|
||||||
delete=False) as f:
|
delete=False) as f:
|
||||||
f.write(doc_data)
|
f.write(doc_data)
|
||||||
os.utime(f.name, times=(t, t))
|
os.utime(f.name, times=(t, t))
|
||||||
|
@ -22,7 +22,7 @@ def path_check(var, directory):
|
|||||||
exists_hint.format(directory)
|
exists_hint.format(directory)
|
||||||
))
|
))
|
||||||
elif not os.access(directory, os.W_OK | os.X_OK):
|
elif not os.access(directory, os.W_OK | os.X_OK):
|
||||||
messages.append(Error(
|
messages.append(Warning(
|
||||||
writeable_message.format(var),
|
writeable_message.format(var),
|
||||||
writeable_hint.format(directory)
|
writeable_hint.format(directory)
|
||||||
))
|
))
|
||||||
|
@ -366,8 +366,10 @@ LOGGING = {
|
|||||||
|
|
||||||
def default_task_workers():
|
def default_task_workers():
|
||||||
# always leave one core open
|
# always leave one core open
|
||||||
available_cores = max(multiprocessing.cpu_count() - 1, 1)
|
available_cores = max(multiprocessing.cpu_count(), 1)
|
||||||
try:
|
try:
|
||||||
|
if available_cores < 4:
|
||||||
|
return available_cores
|
||||||
return max(
|
return max(
|
||||||
math.floor(math.sqrt(available_cores)),
|
math.floor(math.sqrt(available_cores)),
|
||||||
1
|
1
|
||||||
@ -388,7 +390,7 @@ Q_CLUSTER = {
|
|||||||
|
|
||||||
def default_threads_per_worker(task_workers):
|
def default_threads_per_worker(task_workers):
|
||||||
# always leave one core open
|
# always leave one core open
|
||||||
available_cores = max(multiprocessing.cpu_count() - 1, 1)
|
available_cores = max(multiprocessing.cpu_count(), 1)
|
||||||
try:
|
try:
|
||||||
return max(
|
return max(
|
||||||
math.floor(available_cores / task_workers),
|
math.floor(available_cores / task_workers),
|
||||||
|
Loading…
x
Reference in New Issue
Block a user