diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml
index 3f29dbfcb..21cf982d6 100644
--- a/.github/workflows/ci.yml
+++ b/.github/workflows/ci.yml
@@ -37,7 +37,7 @@ jobs:
uses: actions/checkout@v4
-
name: Install python
- uses: actions/setup-python@v4
+ uses: actions/setup-python@v5
with:
python-version: ${{ env.DEFAULT_PYTHON_VERSION }}
-
@@ -56,7 +56,7 @@ jobs:
-
name: Set up Python
id: setup-python
- uses: actions/setup-python@v4
+ uses: actions/setup-python@v5
with:
python-version: ${{ env.DEFAULT_PYTHON_VERSION }}
cache: "pipenv"
@@ -87,7 +87,7 @@ jobs:
pipenv --python ${{ steps.setup-python.outputs.python-version }} run mkdocs gh-deploy --force --no-history
-
name: Upload artifact
- uses: actions/upload-artifact@v3
+ uses: actions/upload-artifact@v4
with:
name: documentation
path: site/
@@ -114,7 +114,7 @@ jobs:
-
name: Set up Python
id: setup-python
- uses: actions/setup-python@v4
+ uses: actions/setup-python@v5
with:
python-version: "${{ matrix.python-version }}"
cache: "pipenv"
@@ -155,7 +155,7 @@ jobs:
-
name: Upload coverage
if: ${{ matrix.python-version == env.DEFAULT_PYTHON_VERSION }}
- uses: actions/upload-artifact@v3
+ uses: actions/upload-artifact@v4
with:
name: backend-coverage-report
path: src/coverage.xml
@@ -238,7 +238,7 @@ jobs:
-
name: Upload Jest coverage
if: always()
- uses: actions/upload-artifact@v3
+ uses: actions/upload-artifact@v4
with:
name: jest-coverage-report-${{ matrix.shard-index }}
path: |
@@ -253,9 +253,9 @@ jobs:
-
name: Upload Playwright test results
if: always()
- uses: actions/upload-artifact@v3
+ uses: actions/upload-artifact@v4
with:
- name: playwright-report
+ name: playwright-report-${{ matrix.shard-index }}
path: src-ui/playwright-report
retention-days: 7
@@ -269,10 +269,18 @@ jobs:
-
uses: actions/checkout@v4
-
- name: Download frontend coverage
- uses: actions/download-artifact@v3
+ name: Download frontend jest coverage
+ uses: actions/download-artifact@v4
with:
path: src-ui/coverage/
+ pattern: jest-coverage-report-*
+ -
+ name: Download frontend playwright coverage
+ uses: actions/download-artifact@v4
+ with:
+ path: src-ui/coverage/
+ pattern: playwright-report-*
+ merge-multiple: true
-
name: Upload frontend coverage to Codecov
uses: codecov/codecov-action@v3
@@ -285,7 +293,7 @@ jobs:
files: '!coverage.xml'
-
name: Download backend coverage
- uses: actions/download-artifact@v3
+ uses: actions/download-artifact@v4
with:
name: backend-coverage-report
path: src/
@@ -416,7 +424,7 @@ jobs:
docker cp frontend-extract:/usr/src/paperless/src/documents/static/frontend src/documents/static/frontend/
-
name: Upload frontend artifact
- uses: actions/upload-artifact@v3
+ uses: actions/upload-artifact@v4
with:
name: frontend-compiled
path: src/documents/static/frontend/
@@ -435,7 +443,7 @@ jobs:
-
name: Set up Python
id: setup-python
- uses: actions/setup-python@v4
+ uses: actions/setup-python@v5
with:
python-version: ${{ env.DEFAULT_PYTHON_VERSION }}
cache: "pipenv"
@@ -461,13 +469,13 @@ jobs:
sudo apt-get install -qq --no-install-recommends gettext liblept5
-
name: Download frontend artifact
- uses: actions/download-artifact@v3
+ uses: actions/download-artifact@v4
with:
name: frontend-compiled
path: src/documents/static/frontend/
-
name: Download documentation artifact
- uses: actions/download-artifact@v3
+ uses: actions/download-artifact@v4
with:
name: documentation
path: docs/_build/html/
@@ -533,7 +541,7 @@ jobs:
tar -cJf paperless-ngx.tar.xz paperless-ngx/
-
name: Upload release artifact
- uses: actions/upload-artifact@v3
+ uses: actions/upload-artifact@v4
with:
name: release
path: dist/paperless-ngx.tar.xz
@@ -552,7 +560,7 @@ jobs:
steps:
-
name: Download release artifact
- uses: actions/download-artifact@v3
+ uses: actions/download-artifact@v4
with:
name: release
path: ./
@@ -603,7 +611,7 @@ jobs:
ref: main
-
name: Set up Python
- uses: actions/setup-python@v4
+ uses: actions/setup-python@v5
with:
python-version: ${{ env.DEFAULT_PYTHON_VERSION }}
cache: "pipenv"
diff --git a/.github/workflows/codeql-analysis.yml b/.github/workflows/codeql-analysis.yml
index bd3943144..fc521843f 100644
--- a/.github/workflows/codeql-analysis.yml
+++ b/.github/workflows/codeql-analysis.yml
@@ -42,7 +42,7 @@ jobs:
# Initializes the CodeQL tools for scanning.
- name: Initialize CodeQL
- uses: github/codeql-action/init@v2
+ uses: github/codeql-action/init@v3
with:
languages: ${{ matrix.language }}
# If you wish to specify custom queries, you can do so here or in a config file.
@@ -51,4 +51,4 @@ jobs:
# queries: ./path/to/local/query, your-org/your-repo/queries@main
- name: Perform CodeQL Analysis
- uses: github/codeql-action/analyze@v2
+ uses: github/codeql-action/analyze@v3
diff --git a/.github/workflows/repo-maintenance.yml b/.github/workflows/repo-maintenance.yml
index c732d0342..ec0153726 100644
--- a/.github/workflows/repo-maintenance.yml
+++ b/.github/workflows/repo-maintenance.yml
@@ -18,7 +18,7 @@ jobs:
name: 'Stale'
runs-on: ubuntu-latest
steps:
- - uses: actions/stale@v8
+ - uses: actions/stale@v9
with:
days-before-stale: 7
days-before-close: 14
diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml
index 86b7a7dc7..16d0b32ea 100644
--- a/.pre-commit-config.yaml
+++ b/.pre-commit-config.yaml
@@ -11,6 +11,8 @@ repos:
- id: check-json
exclude: "tsconfig.*json"
- id: check-yaml
+ args:
+ - "--unsafe"
- id: check-toml
- id: check-executables-have-shebangs
- id: end-of-file-fixer
diff --git a/Dockerfile b/Dockerfile
index 1f9134651..46fdbcd0b 100644
--- a/Dockerfile
+++ b/Dockerfile
@@ -270,3 +270,5 @@ ENTRYPOINT ["/sbin/docker-entrypoint.sh"]
EXPOSE 8000
CMD ["/usr/local/bin/paperless_cmd.sh"]
+
+HEALTHCHECK --interval=30s --timeout=10s --retries=5 CMD [ "curl", "-fs", "-S", "--max-time", "2", "http://localhost:8000" ]
diff --git a/docker/docker-entrypoint.sh b/docker/docker-entrypoint.sh
index 58e46bd01..37f9d7288 100755
--- a/docker/docker-entrypoint.sh
+++ b/docker/docker-entrypoint.sh
@@ -86,13 +86,13 @@ initialize() {
"${CONSUME_DIR}"; do
if [[ ! -d "${dir}" ]]; then
echo "Creating directory ${dir}"
- mkdir "${dir}"
+ mkdir --parents "${dir}"
fi
done
local -r tmp_dir="/tmp/paperless"
echo "Creating directory ${tmp_dir}"
- mkdir -p "${tmp_dir}"
+ mkdir --parents "${tmp_dir}"
set +e
echo "Adjusting permissions of paperless files. This may take a while."
diff --git a/docs/advanced_usage.md b/docs/advanced_usage.md
index ca5ed4321..4bbd7753d 100644
--- a/docs/advanced_usage.md
+++ b/docs/advanced_usage.md
@@ -136,6 +136,11 @@ script can access the following relevant environment variables set:
be triggered, leading to failures as two tasks work on the
same document path
+!!! warning
+
+ If your script modifies `DOCUMENT_WORKING_PATH` in a non-deterministic
+ way, this may allow duplicate documents to be stored
+
A simple but common example for this would be creating a simple script
like this:
diff --git a/docs/api.md b/docs/api.md
index 82244936f..97ccf4c3a 100644
--- a/docs/api.md
+++ b/docs/api.md
@@ -8,7 +8,6 @@ most of the available filters and ordering fields.
The API provides the following main endpoints:
-- `/api/consumption_templates/`: Full CRUD support.
- `/api/correspondents/`: Full CRUD support.
- `/api/custom_fields/`: Full CRUD support.
- `/api/documents/`: Full CRUD support, except POSTing new documents.
@@ -24,6 +23,7 @@ The API provides the following main endpoints:
- `/api/tags/`: Full CRUD support.
- `/api/tasks/`: Read-only.
- `/api/users/`: Full CRUD support.
+- `/api/workflows/`: Full CRUD support.
All of these endpoints except for the logging endpoint allow you to
fetch (and edit and delete where appropriate) individual objects by
@@ -274,6 +274,7 @@ The endpoint supports the following optional form fields:
- `correspondent`: Specify the ID of a correspondent that the consumer
should use for the document.
- `document_type`: Similar to correspondent.
+- `storage_path`: Similar to correspondent.
- `tags`: Similar to correspondent. Specify this multiple times to
have multiple tags added to the document.
- `archive_serial_number`: An optional archive serial number to set.
diff --git a/docs/assets/screenshots/consumption_template.png b/docs/assets/screenshots/consumption_template.png
deleted file mode 100644
index c3802c55f..000000000
Binary files a/docs/assets/screenshots/consumption_template.png and /dev/null differ
diff --git a/docs/assets/screenshots/workflow.png b/docs/assets/screenshots/workflow.png
new file mode 100644
index 000000000..f196e22ba
Binary files /dev/null and b/docs/assets/screenshots/workflow.png differ
diff --git a/docs/configuration.md b/docs/configuration.md
index 6f35be06d..30db90c85 100644
--- a/docs/configuration.md
+++ b/docs/configuration.md
@@ -3,6 +3,11 @@
Paperless provides a wide range of customizations. Depending on how you
run paperless, these settings have to be defined in different places.
+Certain configuration options may be set via the UI. This currently includes
+common [OCR](#ocr) related settings. If set, these will take preference over the
+settings via environment variables. If not set, the environment setting or applicable
+default will be utilized instead.
+
- If you run paperless on docker, `paperless.conf` is not used.
Rather, configure paperless by copying necessary options to
`docker-compose.env`.
diff --git a/docs/index.md b/docs/index.md
index 6dc49bfc6..cd1bd9029 100644
--- a/docs/index.md
+++ b/docs/index.md
@@ -18,6 +18,7 @@ physical documents into a searchable online archive so you can keep, well, _less
## Features
- **Organize and index** your scanned documents with tags, correspondents, types, and more.
+- _Your_ data is stored locally on _your_ server and is never transmitted or shared in any way.
- Performs **OCR** on your documents, adding searchable and selectable text, even to documents scanned with only images.
- Utilizes the open-source Tesseract engine to recognize more than 100 languages.
- Documents are saved as PDF/A format which is designed for long term storage, alongside the unaltered originals.
@@ -41,7 +42,7 @@ physical documents into a searchable online archive so you can keep, well, _less
- Configure multiple accounts and rules for each account.
- After processing, paperless can perform actions on the messages such as marking as read, deleting and more.
- A built-in robust **multi-user permissions** system that supports 'global' permissions as well as per document or object.
-- A powerful templating system that gives you more control over the consumption pipeline.
+- A powerful workflow system that gives you even more control.
- **Optimized** for multi core systems: Paperless-ngx consumes multiple documents in parallel.
- The integrated sanity checker makes sure that your document archive is in good health.
@@ -156,9 +157,9 @@ Tag, correspondent, document type and storage path editing.
- Consumption templates provide finer control over the document pipeline.
+ Workflows provide finer control over the document pipeline and trigger actions.
-
+
diff --git a/docs/setup.md b/docs/setup.md
index e1b8db13e..4ca0085b7 100644
--- a/docs/setup.md
+++ b/docs/setup.md
@@ -28,7 +28,8 @@ steps described in [Docker setup](#docker_hub) automatically.
1. Make sure that Docker and Docker Compose are installed.
!!! tip
- See the Docker installation instructions at https://docs.docker.com/engine/install/
+
+ See the Docker installation instructions at https://docs.docker.com/engine/install/
2. Download and run the installation script:
@@ -72,7 +73,7 @@ steps described in [Docker setup](#docker_hub) automatically.
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
- version **v2**. To check do: `docker compose -v` or `docker -v`
+ version **v2**. To check do: `docker compose version` or `docker -v`
See the [Docker installation guide](https://docs.docker.com/engine/install/) on how to install the current
version of Docker for your operating system or Linux distribution of
diff --git a/docs/usage.md b/docs/usage.md
index 42701728d..7e8db4acf 100644
--- a/docs/usage.md
+++ b/docs/usage.md
@@ -238,7 +238,7 @@ do not have an owner set.
### Default permissions
-Default permissions for documents can be set using consumption templates.
+Default permissions for documents can be set using workflows.
For objects created via the web UI (tags, doc types, etc.) the default is to set the current user
as owner and no extra permissions, but you explicitly set these under Settings > Permissions.
@@ -255,29 +255,80 @@ permissions can be granted to limit access to certain parts of the UI (and corre
In order to enable the password reset feature you will need to setup an SMTP backend, see
[`PAPERLESS_EMAIL_HOST`](configuration.md#PAPERLESS_EMAIL_HOST)
-## Consumption templates
+## Workflows
-Consumption templates were introduced in v2.0 and allow for finer control over what metadata (tags, doc
-types) and permissions (owner, privileges) are assigned to documents during consumption. In general,
-templates are applied sequentially (by sort order) but subsequent templates will never override an
-assignment from a preceding template. The same is true for mail rules, e.g. if you set the correspondent
-in a mail rule any subsequent consumption templates that are applied _will not_ overwrite this. The
-exception to this is assignments that can be multiple e.g. tags and permissions, which will be merged.
+!!! note
-Consumption templates allow you to filter by:
+ v2.3 added "Workflows" and existing "Consumption Templates" were converted automatically to the new more powerful format.
+
+Workflows allow hooking into the Paperless-ngx document pipeline, for example to alter what metadata (tags, doc types) and
+permissions (owner, privileges) are assigned to documents. Workflows can have multiple 'triggers' and 'actions'. Triggers
+are events (with optional filtering rules) that will cause the workflow to be run and actions are the set of sequential
+actions to apply.
+
+In general, workflows and any actions they contain are applied sequentially by sort order. For "assignment" actions, subsequent
+workflow actions will override previous assignments, except for assignments that accept multiple items e.g. tags, custom
+fields and permissions, which will be merged.
+
+### Workflow Triggers
+
+Currently, there are three events that correspond to workflow trigger 'types':
+
+1. **Consumption Started**: _before_ a document is consumed, so events can include filters by source (mail, consumption
+ folder or API), file path, file name, mail rule
+2. **Document Added**: _after_ a document is added. At this time, file path and source information is no longer available,
+ but the document content has been extracted and metadata such as document type, tags, etc. have been set, so these can now
+ be used for filtering.
+3. **Document Updated**: when a document is updated. Similar to 'added' events, triggers can include filtering by content matching,
+ tags, doc type, or correspondent.
+
+The following flow diagram illustrates the three trigger types:
+
+```mermaid
+flowchart TD
+ consumption{"Matching
+ 'Consumption'
+ trigger(s)"}
+
+ added{"Matching
+ 'Added'
+ trigger(s)"}
+
+ updated{"Matching
+ 'Updated'
+ trigger(s)"}
+
+ A[New Document] --> consumption
+ consumption --> |Yes| C[Workflow Actions Run]
+ consumption --> |No| D
+ C --> D[Document Added]
+ D -- Paperless-ngx 'matching' of tags, etc. --> added
+ added --> |Yes| F[Workflow Actions Run]
+ added --> |No| G
+ F --> G[Document Finalized]
+ H[Existing Document Changed] --> updated
+ updated --> |Yes| J[Workflow Actions Run]
+ updated --> |No| K
+ J --> K[Document Saved]
+```
+
+#### Filters {#workflow-trigger-filters}
+
+Workflows allow you to filter by:
- Source, e.g. documents uploaded via consume folder, API (& the web UI) and mail fetch
- File name, including wildcards e.g. \*.pdf will apply to all pdfs
- File path, including wildcards. Note that enabling `PAPERLESS_CONSUMER_RECURSIVE` would allow, for
example, automatically assigning documents to different owners based on the upload directory.
-- Mail rule. Choosing this option will force 'mail fetch' to be the template source.
+- Mail rule. Choosing this option will force 'mail fetch' to be the workflow source.
+- Content matching (`Added` and `Updated` triggers only). Filter document content using the matching settings.
+- Tags (`Added` and `Updated` triggers only). Filter for documents with any of the specified tags
+- Document type (`Added` and `Updated` triggers only). Filter documents with this doc type
+- Correspondent (`Added` and `Updated` triggers only). Filter documents with this correspondent
-!!! note
+### Workflow Actions
- You must include a file name filter, a path filter or a mail rule filter. Use * for either to apply
- to all files.
-
-Consumption templates can assign:
+There is currently one type of workflow action, "Assignment", which can assign:
- Title, see [title placeholders](usage.md#title-placeholders) below
- Tags, correspondent, document types
@@ -285,21 +336,11 @@ Consumption templates can assign:
- View and / or edit permissions to users or groups
- Custom fields. Note that no value for the field will be set
-### Consumption template permissions
+#### Title placeholders
-All users who have application permissions for editing consumption templates can see the same set
-of templates. In other words, templates themselves intentionally do not have an owner or permissions.
-
-Given their potentially far-reaching capabilities, you may want to restrict access to templates.
-
-Upon migration, existing installs will grant access to consumption templates to users who can add
-documents (and superusers who can always access all parts of the app).
-
-### Title placeholders
-
-Consumption template titles can include placeholders, _only for items that are assigned within the template_.
-This is because at the time of consumption (when the title is to be set), no automatic tags etc. have been
-applied. You can use the following placeholders:
+Workflow titles can include placeholders but the available options differ depending on the type of
+workflow trigger. This is because at the time of consumption (when the title is to be set), no automatic tags etc. have been
+applied. You can use the following placeholders with any trigger type:
- `{correspondent}`: assigned correspondent name
- `{document_type}`: assigned document type name
@@ -314,6 +355,27 @@ applied. You can use the following placeholders:
- `{added_time}`: added time in HH:MM format
- `{original_filename}`: original file name without extension
+The following placeholders are only available for "added" or "updated" triggers
+
+- `{created}`: created datetime
+- `{created_year}`: created year
+- `{created_year_short}`: created year
+- `{created_month}`: created month
+- `{created_month_name}`: created month name
+- `{created_month_name_short}`: created month short name
+- `{created_day}`: created day
+- `{created_time}`: created time in HH:MM format
+
+### Workflow permissions
+
+All users who have application permissions for editing workflows can see the same set
+of workflows. In other words, workflows themselves intentionally do not have an owner or permissions.
+
+Given their potentially far-reaching capabilities, you may want to restrict access to workflows.
+
+Upon migration, existing installs will grant access to workflows to users who can add
+documents (and superusers who can always access all parts of the app).
+
## Custom Fields {#custom-fields}
Paperless-ngx supports the use of custom fields for documents as of v2.0, allowing a user
diff --git a/mkdocs.yml b/mkdocs.yml
index 816ba932c..8e1e822af 100644
--- a/mkdocs.yml
+++ b/mkdocs.yml
@@ -44,6 +44,11 @@ markdown_extensions:
- pymdownx.inlinehilite
- pymdownx.snippets
- footnotes
+ - pymdownx.superfences:
+ custom_fences:
+ - name: mermaid
+ class: mermaid
+ format: !!python/name:pymdownx.superfences.fence_code_format
strict: true
nav:
- index.md
diff --git a/src-ui/messages.xlf b/src-ui/messages.xlf
index 0f28ee783..7e45a4550 100644
--- a/src-ui/messages.xlf
+++ b/src-ui/messages.xlf
@@ -346,8 +346,8 @@
172
-
- Consumption templates give you finer control over the document ingestion process.
+
+ Workflows give you more control over the document pipeline.src/app/app.component.ts180
@@ -388,6 +388,156 @@
208
+
+ Configuration
+
+ src/app/components/admin/config/config.component.html
+ 1
+
+
+ src/app/components/app-frame/app-frame.component.html
+ 276
+
+
+ src/app/components/app-frame/app-frame.component.html
+ 280
+
+
+
+
+
+ src/app/components/admin/config/config.component.html
+ 8,9
+
+
+ src/app/components/admin/tasks/tasks.component.html
+ 11
+
+
+ src/app/components/common/input/tags/tags.component.html
+ 4
+
+
+ src/app/components/common/permissions-select/permissions-select.component.html
+ 22
+
+
+
+ Read the documentation about this setting
+
+ src/app/components/admin/config/config.component.html
+ 19
+
+
+
+ Enable
+
+ src/app/components/admin/config/config.component.html
+ 30
+
+
+
+ Discard
+
+ src/app/components/admin/config/config.component.html
+ 48
+
+
+ src/app/components/document-detail/document-detail.component.html
+ 339
+
+
+
+ Save
+
+ src/app/components/admin/config/config.component.html
+ 51
+
+
+ src/app/components/admin/settings/settings.component.html
+ 337
+
+
+ src/app/components/common/edit-dialog/correspondent-edit-dialog/correspondent-edit-dialog.component.html
+ 25
+
+
+ src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.html
+ 16
+
+
+ src/app/components/common/edit-dialog/document-type-edit-dialog/document-type-edit-dialog.component.html
+ 27
+
+
+ src/app/components/common/edit-dialog/group-edit-dialog/group-edit-dialog.component.html
+ 17
+
+
+ src/app/components/common/edit-dialog/mail-account-edit-dialog/mail-account-edit-dialog.component.html
+ 37
+
+
+ src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html
+ 49
+
+
+ src/app/components/common/edit-dialog/storage-path-edit-dialog/storage-path-edit-dialog.component.html
+ 26
+
+
+ src/app/components/common/edit-dialog/tag-edit-dialog/tag-edit-dialog.component.html
+ 28
+
+
+ src/app/components/common/edit-dialog/user-edit-dialog/user-edit-dialog.component.html
+ 36
+
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 171
+
+
+ src/app/components/common/profile-edit-dialog/profile-edit-dialog.component.html
+ 58
+
+
+ src/app/components/document-detail/document-detail.component.html
+ 331
+
+
+ src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html
+ 21
+
+
+
+ Error retrieving config
+
+ src/app/components/admin/config/config.component.ts
+ 79
+
+
+
+ Invalid JSON
+
+ src/app/components/admin/config/config.component.ts
+ 105
+
+
+
+ Configuration updated
+
+ src/app/components/admin/config/config.component.ts
+ 148
+
+
+
+ An error occurred updating configuration
+
+ src/app/components/admin/config/config.component.ts
+ 153
+
+ Logs
@@ -396,11 +546,11 @@
src/app/components/app-frame/app-frame.component.html
- 300
+ 309src/app/components/app-frame/app-frame.component.html
- 305
+ 314
@@ -458,7 +608,7 @@
src/app/components/document-detail/document-detail.component.html
- 303
+ 348src/app/components/document-list/document-list.component.html
@@ -767,7 +917,7 @@
src/app/components/document-detail/document-detail.component.html
- 252
+ 297src/app/components/document-list/bulk-editor/bulk-editor.component.html
@@ -852,12 +1002,12 @@
236
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 47
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 116
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 66
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 135src/app/components/common/input/permissions/permissions-form/permissions-form.component.html
@@ -879,12 +1029,12 @@
246
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 55
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 124
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 74
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 143src/app/components/common/input/permissions/permissions-form/permissions-form.component.html
@@ -909,8 +1059,8 @@
255
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 80
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 149src/app/components/common/input/permissions/permissions-form/permissions-form.component.html
@@ -1013,10 +1163,6 @@
src/app/components/admin/users-groups/users-groups.component.html63
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 10
- src/app/components/common/edit-dialog/correspondent-edit-dialog/correspondent-edit-dialog.component.html9
@@ -1050,12 +1196,12 @@
8
- src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html
- 8
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 10
- src/app/components/manage/consumption-templates/consumption-templates.component.html
- 14
+ src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html
+ 8src/app/components/manage/custom-fields/custom-fields.component.html
@@ -1101,6 +1247,10 @@
src/app/components/manage/management-list/management-list.component.html41
+
+ src/app/components/manage/workflows/workflows.component.html
+ 14
+ Appears on
@@ -1149,6 +1299,10 @@
src/app/components/admin/users-groups/users-groups.component.html66
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 66
+ src/app/components/document-detail/document-detail.component.html49
@@ -1157,10 +1311,6 @@
src/app/components/document-list/bulk-editor/bulk-editor.component.html86
-
- src/app/components/manage/consumption-templates/consumption-templates.component.html
- 17
- src/app/components/manage/custom-fields/custom-fields.component.html16
@@ -1189,6 +1339,10 @@
src/app/components/manage/management-list/management-list.component.html47
+
+ src/app/components/manage/workflows/workflows.component.html
+ 18
+ Delete
@@ -1208,6 +1362,14 @@
src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.ts53
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 48
+
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 92
+ src/app/components/common/permissions-select/permissions-select.component.html9
@@ -1224,10 +1386,6 @@
src/app/components/document-list/bulk-editor/bulk-editor.component.html142
-
- src/app/components/manage/consumption-templates/consumption-templates.component.html
- 37
- src/app/components/manage/custom-fields/custom-fields.component.html35
@@ -1276,6 +1434,10 @@
src/app/components/manage/management-list/management-list.component.ts205
+
+ src/app/components/manage/workflows/workflows.component.html
+ 39
+ No saved views defined.
@@ -1284,65 +1446,6 @@
319
-
- Save
-
- src/app/components/admin/settings/settings.component.html
- 337
-
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 93
-
-
- src/app/components/common/edit-dialog/correspondent-edit-dialog/correspondent-edit-dialog.component.html
- 25
-
-
- src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.html
- 16
-
-
- src/app/components/common/edit-dialog/document-type-edit-dialog/document-type-edit-dialog.component.html
- 27
-
-
- src/app/components/common/edit-dialog/group-edit-dialog/group-edit-dialog.component.html
- 17
-
-
- src/app/components/common/edit-dialog/mail-account-edit-dialog/mail-account-edit-dialog.component.html
- 37
-
-
- src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html
- 49
-
-
- src/app/components/common/edit-dialog/storage-path-edit-dialog/storage-path-edit-dialog.component.html
- 26
-
-
- src/app/components/common/edit-dialog/tag-edit-dialog/tag-edit-dialog.component.html
- 28
-
-
- src/app/components/common/edit-dialog/user-edit-dialog/user-edit-dialog.component.html
- 36
-
-
- src/app/components/common/profile-edit-dialog/profile-edit-dialog.component.html
- 58
-
-
- src/app/components/document-detail/document-detail.component.html
- 286
-
-
- src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html
- 21
-
- Use system language
@@ -1433,7 +1536,7 @@
src/app/components/app-frame/app-frame.component.html
- 287
+ 296
@@ -1459,21 +1562,6 @@
5
-
-
-
- src/app/components/admin/tasks/tasks.component.html
- 11
-
-
- src/app/components/common/input/tags/tags.component.html
- 4
-
-
- src/app/components/common/permissions-select/permissions-select.component.html
- 22
-
- Created
@@ -1635,11 +1723,11 @@
src/app/components/app-frame/app-frame.component.html
- 276
+ 285src/app/components/app-frame/app-frame.component.html
- 280
+ 289
@@ -1716,10 +1804,6 @@
src/app/components/document-list/document-card-small/document-card-small.component.html105
-
- src/app/components/manage/consumption-templates/consumption-templates.component.html
- 32
- src/app/components/manage/custom-fields/custom-fields.component.html30
@@ -1764,6 +1848,10 @@
src/app/components/manage/management-list/management-list.component.html103
+
+ src/app/components/manage/workflows/workflows.component.html
+ 34
+ Add Group
@@ -1840,10 +1928,6 @@
src/app/components/document-list/bulk-editor/bulk-editor.component.ts500
-
- src/app/components/manage/consumption-templates/consumption-templates.component.ts
- 91
- src/app/components/manage/custom-fields/custom-fields.component.ts73
@@ -1856,6 +1940,10 @@
src/app/components/manage/mail/mail.component.ts173
+
+ src/app/components/manage/workflows/workflows.component.ts
+ 97
+ Proceed
@@ -1875,10 +1963,6 @@
src/app/components/document-list/bulk-editor/bulk-editor.component.ts502
-
- src/app/components/manage/consumption-templates/consumption-templates.component.ts
- 93
- src/app/components/manage/custom-fields/custom-fields.component.ts75
@@ -1891,6 +1975,10 @@
src/app/components/manage/mail/mail.component.ts175
+
+ src/app/components/manage/workflows/workflows.component.ts
+ 99
+ Deleted user
@@ -1992,11 +2080,11 @@
src/app/components/app-frame/app-frame.component.html
- 310
+ 319src/app/components/app-frame/app-frame.component.html
- 315
+ 324
@@ -2165,19 +2253,20 @@
1
-
- Consumption templates
+
+ Workflowssrc/app/components/app-frame/app-frame.component.html241
-
-
- Templatessrc/app/components/app-frame/app-frame.component.html245
+
+ src/app/components/manage/workflows/workflows.component.html
+ 1
+ Mail
@@ -2201,49 +2290,49 @@
File Taskssrc/app/components/app-frame/app-frame.component.html
- 294,296
+ 303,305GitHubsrc/app/components/app-frame/app-frame.component.html
- 322
+ 331is available.src/app/components/app-frame/app-frame.component.html
- 331,332
+ 340,341Click to view.src/app/components/app-frame/app-frame.component.html
- 332
+ 341Paperless-ngx can automatically check for updatessrc/app/components/app-frame/app-frame.component.html
- 336
+ 345 How does this work? src/app/components/app-frame/app-frame.component.html
- 343,345
+ 352,354Update availablesrc/app/components/app-frame/app-frame.component.html
- 359
+ 368
@@ -2426,278 +2515,6 @@
57
-
- Sort order
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 13
-
-
- src/app/components/manage/consumption-templates/consumption-templates.component.html
- 15
-
-
-
- Filters
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 18
-
-
-
- Process documents that match all filters specified below.
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 19
-
-
-
- Filter sources
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 20
-
-
-
- Filter filename
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 21
-
-
-
- Apply to documents that match this filename. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive.
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 21
-
-
-
- Filter path
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 22
-
-
-
- Apply to documents that match this path. Wildcards specified as * are allowed. Case insensitive.</a>
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 22
-
-
-
- Filter mail rule
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 23
-
-
-
- Apply to documents consumed via this mail rule.
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 23
-
-
-
- Assignments
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 28
-
-
-
- Assign title
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 33
-
-
-
- Can include some placeholders, see <a target='_blank' href='https://docs.paperless-ngx.com/usage/#consumption-templates'>documentation</a>.
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 33
-
-
-
- Assign tags
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 34
-
-
-
- Assign document type
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 35
-
-
- src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html
- 35
-
-
-
- Assign correspondent
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 36
-
-
- src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html
- 38
-
-
-
- Assign storage path
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 37
-
-
-
- Assign custom fields
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 38
-
-
-
- Assign owner
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 41
-
-
-
- Assign view permissions
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 43
-
-
-
- Assign edit permissions
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 62
-
-
-
- Error
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 90
-
-
- src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html
- 46
-
-
- src/app/components/common/toasts/toasts.component.html
- 30
-
-
-
- Cancel
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html
- 92
-
-
- src/app/components/common/edit-dialog/correspondent-edit-dialog/correspondent-edit-dialog.component.html
- 24
-
-
- src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.html
- 15
-
-
- src/app/components/common/edit-dialog/document-type-edit-dialog/document-type-edit-dialog.component.html
- 26
-
-
- src/app/components/common/edit-dialog/group-edit-dialog/group-edit-dialog.component.html
- 16
-
-
- src/app/components/common/edit-dialog/mail-account-edit-dialog/mail-account-edit-dialog.component.html
- 36
-
-
- src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html
- 48
-
-
- src/app/components/common/edit-dialog/storage-path-edit-dialog/storage-path-edit-dialog.component.html
- 25
-
-
- src/app/components/common/edit-dialog/tag-edit-dialog/tag-edit-dialog.component.html
- 27
-
-
- src/app/components/common/edit-dialog/user-edit-dialog/user-edit-dialog.component.html
- 35
-
-
- src/app/components/common/permissions-dialog/permissions-dialog.component.html
- 22
-
-
- src/app/components/common/profile-edit-dialog/profile-edit-dialog.component.html
- 57
-
-
- src/app/components/common/select-dialog/select-dialog.component.html
- 12
-
-
- src/app/components/document-list/bulk-editor/bulk-editor.component.html
- 6
-
-
- src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html
- 20
-
-
-
- Consume Folder
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.ts
- 27
-
-
-
- API Upload
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.ts
- 31
-
-
-
- Mail Fetch
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.ts
- 35
-
-
-
- Create new consumption template
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.ts
- 92
-
-
-
- Edit consumption template
-
- src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.ts
- 96
-
- Matching algorithm
@@ -2754,6 +2571,73 @@
src/app/components/common/edit-dialog/tag-edit-dialog/tag-edit-dialog.component.html18
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 194
+
+
+
+ Cancel
+
+ src/app/components/common/edit-dialog/correspondent-edit-dialog/correspondent-edit-dialog.component.html
+ 24
+
+
+ src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.html
+ 15
+
+
+ src/app/components/common/edit-dialog/document-type-edit-dialog/document-type-edit-dialog.component.html
+ 26
+
+
+ src/app/components/common/edit-dialog/group-edit-dialog/group-edit-dialog.component.html
+ 16
+
+
+ src/app/components/common/edit-dialog/mail-account-edit-dialog/mail-account-edit-dialog.component.html
+ 36
+
+
+ src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html
+ 48
+
+
+ src/app/components/common/edit-dialog/storage-path-edit-dialog/storage-path-edit-dialog.component.html
+ 25
+
+
+ src/app/components/common/edit-dialog/tag-edit-dialog/tag-edit-dialog.component.html
+ 27
+
+
+ src/app/components/common/edit-dialog/user-edit-dialog/user-edit-dialog.component.html
+ 35
+
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 170
+
+
+ src/app/components/common/permissions-dialog/permissions-dialog.component.html
+ 22
+
+
+ src/app/components/common/profile-edit-dialog/profile-edit-dialog.component.html
+ 57
+
+
+ src/app/components/common/select-dialog/select-dialog.component.html
+ 12
+
+
+ src/app/components/document-list/bulk-editor/bulk-editor.component.html
+ 6
+
+
+ src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html
+ 20
+ Create new correspondent
@@ -3110,6 +2994,17 @@
33
+
+ Assign document type
+
+ src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html
+ 35
+
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 104
+
+ Assign correspondent from
@@ -3117,6 +3012,17 @@
36
+
+ Assign correspondent
+
+ src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html
+ 38
+
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 105
+
+ Assign owner from rule
@@ -3124,6 +3030,21 @@
40
+
+ Error
+
+ src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html
+ 46
+
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 168
+
+
+ src/app/components/common/toasts/toasts.component.html
+ 30
+
+ Only process attachments
@@ -3400,6 +3321,291 @@
48
+
+ Sort order
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 13
+
+
+ src/app/components/manage/workflows/workflows.component.html
+ 15
+
+
+
+ Enabled
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 16
+
+
+ src/app/components/manage/workflows/workflows.component.html
+ 27
+
+
+
+ Triggers
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 22
+
+
+ src/app/components/manage/workflows/workflows.component.html
+ 17
+
+
+
+ Trigger Workflow On:
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 28
+
+
+
+ Add Trigger
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 33
+
+
+
+ Apply Actions:
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 72
+
+
+
+ Add Action
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 77
+
+
+
+ Action type
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 98
+
+
+
+ Assign title
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 102
+
+
+
+ Can include some placeholders, see <a target='_blank' href='https://docs.paperless-ngx.com/usage/#workflows'>documentation</a>.
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 102
+
+
+
+ Assign tags
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 103
+
+
+
+ Assign storage path
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 106
+
+
+
+ Assign custom fields
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 107
+
+
+
+ Assign owner
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 110
+
+
+
+ Assign view permissions
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 112
+
+
+
+ Assign edit permissions
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 131
+
+
+
+ Trigger type
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 178
+
+
+
+ Trigger for documents that match all filters specified below.
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 179
+
+
+
+ Filter filename
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 182
+
+
+
+ Apply to documents that match this filename. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive.
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 182
+
+
+
+ Filter sources
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 184
+
+
+
+ Filter path
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 185
+
+
+
+ Apply to documents that match this path. Wildcards specified as * are allowed. Case insensitive.</a>
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 185
+
+
+
+ Filter mail rule
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 186
+
+
+
+ Apply to documents consumed via this mail rule.
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 186
+
+
+
+ Content matching algorithm
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 189
+
+
+
+ Content matching pattern
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 191
+
+
+
+ Has tags
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 200
+
+
+
+ Has correspondent
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 201
+
+
+
+ Has document type
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html
+ 202
+
+
+
+ Consume Folder
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.ts
+ 38
+
+
+
+ API Upload
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.ts
+ 42
+
+
+
+ Mail Fetch
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.ts
+ 46
+
+
+
+ Consumption Started
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.ts
+ 53
+
+
+
+ Document Added
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.ts
+ 57
+
+
+
+ Document Updated
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.ts
+ 61
+
+
+
+ Assignment
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.ts
+ 68
+
+
+
+ Create new workflow
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.ts
+ 136
+
+
+
+ Edit workflow
+
+ src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.ts
+ 140
+
+ All
@@ -3482,15 +3688,19 @@
src/app/components/common/input/number/number.component.html
- 9
+ 11src/app/components/common/input/select/select.component.html11
+
+ src/app/components/common/input/switch/switch.component.html
+ 10
+ src/app/components/common/input/text/text.component.html
- 9
+ 11src/app/components/common/input/url/url.component.html
@@ -3949,6 +4159,10 @@
src/app/components/common/toasts/toasts.component.html28
+
+ src/app/components/manage/workflows/workflows.component.html
+ 16
+ Copy Raw Error
@@ -4397,14 +4611,14 @@
Contentsrc/app/components/document-detail/document-detail.component.html
- 161
+ 206Metadatasrc/app/components/document-detail/document-detail.component.html
- 170
+ 215src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts
@@ -4415,123 +4629,116 @@
Date modifiedsrc/app/components/document-detail/document-detail.component.html
- 177
+ 222Date addedsrc/app/components/document-detail/document-detail.component.html
- 181
+ 226Media filenamesrc/app/components/document-detail/document-detail.component.html
- 185
+ 230Original filenamesrc/app/components/document-detail/document-detail.component.html
- 189
+ 234Original MD5 checksumsrc/app/components/document-detail/document-detail.component.html
- 193
+ 238Original file sizesrc/app/components/document-detail/document-detail.component.html
- 197
+ 242Original mime typesrc/app/components/document-detail/document-detail.component.html
- 201
+ 246Archive MD5 checksumsrc/app/components/document-detail/document-detail.component.html
- 206
+ 251Archive file sizesrc/app/components/document-detail/document-detail.component.html
- 212
+ 257Original document metadatasrc/app/components/document-detail/document-detail.component.html
- 221
+ 266Archived document metadatasrc/app/components/document-detail/document-detail.component.html
- 224
+ 269Previewsrc/app/components/document-detail/document-detail.component.html
- 231
+ 276Notes src/app/components/document-detail/document-detail.component.html
- 241,244
+ 286,289Enter Passwordsrc/app/components/document-detail/document-detail.component.html
- 275
+ 320src/app/components/document-detail/document-detail.component.html
- 332
+ 377Save & nextsrc/app/components/document-detail/document-detail.component.html
- 288
+ 333Save & closesrc/app/components/document-detail/document-detail.component.html
- 291
-
-
-
- Discard
-
- src/app/components/document-detail/document-detail.component.html
- 294
+ 336
@@ -5505,76 +5712,6 @@
88
-
- Consumption Templates
-
- src/app/components/manage/consumption-templates/consumption-templates.component.html
- 1
-
-
-
- Add Template
-
- src/app/components/manage/consumption-templates/consumption-templates.component.html
- 6
-
-
-
- Document Sources
-
- src/app/components/manage/consumption-templates/consumption-templates.component.html
- 16
-
-
-
- No templates defined.
-
- src/app/components/manage/consumption-templates/consumption-templates.component.html
- 45
-
-
-
- Saved template "".
-
- src/app/components/manage/consumption-templates/consumption-templates.component.ts
- 73
-
-
-
- Error saving template.
-
- src/app/components/manage/consumption-templates/consumption-templates.component.ts
- 81
-
-
-
- Confirm delete template
-
- src/app/components/manage/consumption-templates/consumption-templates.component.ts
- 89
-
-
-
- This operation will permanently delete this template.
-
- src/app/components/manage/consumption-templates/consumption-templates.component.ts
- 90
-
-
-
- Deleted template
-
- src/app/components/manage/consumption-templates/consumption-templates.component.ts
- 99
-
-
-
- Error deleting template.
-
- src/app/components/manage/consumption-templates/consumption-templates.component.ts
- 104
-
- correspondent
@@ -6072,6 +6209,69 @@
53
+
+ Add Workflow
+
+ src/app/components/manage/workflows/workflows.component.html
+ 6
+
+
+
+ Disabled
+
+ src/app/components/manage/workflows/workflows.component.html
+ 27
+
+
+
+ No workflows defined.
+
+ src/app/components/manage/workflows/workflows.component.html
+ 47
+
+
+
+ Saved workflow "".
+
+ src/app/components/manage/workflows/workflows.component.ts
+ 79
+
+
+
+ Error saving workflow.
+
+ src/app/components/manage/workflows/workflows.component.ts
+ 87
+
+
+
+ Confirm delete workflow
+
+ src/app/components/manage/workflows/workflows.component.ts
+ 95
+
+
+
+ This operation will permanently delete this workflow.
+
+ src/app/components/manage/workflows/workflows.component.ts
+ 96
+
+
+
+ Deleted workflow
+
+ src/app/components/manage/workflows/workflows.component.ts
+ 105
+
+
+
+ Error deleting workflow.
+
+ src/app/components/manage/workflows/workflows.component.ts
+ 110
+
+ Not Found
@@ -6226,6 +6426,104 @@
46
+
+ OCR Settings
+
+ src/app/data/paperless-config.ts
+ 49
+
+
+
+ Output Type
+
+ src/app/data/paperless-config.ts
+ 73
+
+
+
+ Language
+
+ src/app/data/paperless-config.ts
+ 81
+
+
+
+ Pages
+
+ src/app/data/paperless-config.ts
+ 88
+
+
+
+ Mode
+
+ src/app/data/paperless-config.ts
+ 95
+
+
+
+ Skip Archive File
+
+ src/app/data/paperless-config.ts
+ 103
+
+
+
+ Image DPI
+
+ src/app/data/paperless-config.ts
+ 111
+
+
+
+ Clean
+
+ src/app/data/paperless-config.ts
+ 118
+
+
+
+ Deskew
+
+ src/app/data/paperless-config.ts
+ 126
+
+
+
+ Rotate Pages
+
+ src/app/data/paperless-config.ts
+ 133
+
+
+
+ Rotate Pages Threshold
+
+ src/app/data/paperless-config.ts
+ 140
+
+
+
+ Max Image Pixels
+
+ src/app/data/paperless-config.ts
+ 147
+
+
+
+ Color Conversion Strategy
+
+ src/app/data/paperless-config.ts
+ 154
+
+
+
+ OCR Arguments
+
+ src/app/data/paperless-config.ts
+ 162
+
+ Warning: You have unsaved changes to your document(s).
diff --git a/src-ui/package-lock.json b/src-ui/package-lock.json
index 030902b88..6a8a18a12 100644
--- a/src-ui/package-lock.json
+++ b/src-ui/package-lock.json
@@ -10,14 +10,14 @@
"hasInstallScript": true,
"dependencies": {
"@angular/cdk": "^17.0.4",
- "@angular/common": "~17.0.7",
- "@angular/compiler": "~17.0.7",
- "@angular/core": "~17.0.7",
- "@angular/forms": "~17.0.7",
- "@angular/localize": "~17.0.7",
- "@angular/platform-browser": "~17.0.7",
- "@angular/platform-browser-dynamic": "~17.0.7",
- "@angular/router": "~17.0.7",
+ "@angular/common": "~17.0.8",
+ "@angular/compiler": "~17.0.8",
+ "@angular/core": "~17.0.8",
+ "@angular/forms": "~17.0.8",
+ "@angular/localize": "~17.0.8",
+ "@angular/platform-browser": "~17.0.8",
+ "@angular/platform-browser-dynamic": "~17.0.8",
+ "@angular/router": "~17.0.8",
"@ng-bootstrap/ng-bootstrap": "^16.0.0",
"@ng-select/ng-select": "^12.0.4",
"@ngneat/dirty-check-forms": "^3.0.3",
@@ -37,21 +37,21 @@
},
"devDependencies": {
"@angular-builders/jest": "17.0.0",
- "@angular-devkit/build-angular": "~17.0.7",
+ "@angular-devkit/build-angular": "~17.0.8",
"@angular-eslint/builder": "17.1.1",
"@angular-eslint/eslint-plugin": "17.1.1",
"@angular-eslint/eslint-plugin-template": "17.1.1",
"@angular-eslint/schematics": "17.1.1",
"@angular-eslint/template-parser": "17.1.1",
- "@angular/cli": "~17.0.7",
+ "@angular/cli": "~17.0.8",
"@angular/compiler-cli": "~17.0.7",
"@playwright/test": "^1.40.1",
"@types/jest": "^29.5.10",
- "@types/node": "^20.10.2",
- "@typescript-eslint/eslint-plugin": "^6.10.0",
- "@typescript-eslint/parser": "^6.10.0",
+ "@types/node": "^20.10.6",
+ "@typescript-eslint/eslint-plugin": "^6.17.0",
+ "@typescript-eslint/parser": "^6.17.0",
"concurrently": "^8.2.2",
- "eslint": "^8.53.0",
+ "eslint": "^8.56.0",
"jest": "29.7.0",
"jest-environment-jsdom": "^29.7.0",
"jest-preset-angular": "^13.1.4",
@@ -107,12 +107,12 @@
}
},
"node_modules/@angular-devkit/architect": {
- "version": "0.1700.7",
- "resolved": "https://registry.npmjs.org/@angular-devkit/architect/-/architect-0.1700.7.tgz",
- "integrity": "sha512-32uitQKsYLGXAKoXBsmOnPsTt9pS+b9cnFI9ZvBFVhJ31I2EOM7vGcMFalhTxdB/DkVHk4TyO78efV0V26DwCA==",
+ "version": "0.1700.8",
+ "resolved": "https://registry.npmjs.org/@angular-devkit/architect/-/architect-0.1700.8.tgz",
+ "integrity": "sha512-SWVr3CvwO6T0yW2ytszCwBT1g92vyFkwbVUxqE93urYnoD8PvP+81GH5YwVjHQTgvhP4eXQMGZ9hpHx57VOrWQ==",
"dev": true,
"dependencies": {
- "@angular-devkit/core": "17.0.7",
+ "@angular-devkit/core": "17.0.8",
"rxjs": "7.8.1"
},
"engines": {
@@ -122,15 +122,15 @@
}
},
"node_modules/@angular-devkit/build-angular": {
- "version": "17.0.7",
- "resolved": "https://registry.npmjs.org/@angular-devkit/build-angular/-/build-angular-17.0.7.tgz",
- "integrity": "sha512-AtEzLk6n6BXqQzk0Bsupe6GV0IgUe7RbpBfqROi+NZqMA7OUAHCX3xA6M68Qu+5KxBtW7T5lHeZZ7iP/y39wtQ==",
+ "version": "17.0.8",
+ "resolved": "https://registry.npmjs.org/@angular-devkit/build-angular/-/build-angular-17.0.8.tgz",
+ "integrity": "sha512-u7R5yX92ZxOL/LfxiKGGqlBo86100sJ5Rabavn8DeGtYP8N0qgwCcNwlW2zaMoUlkw2geMnxcxIX5VJI4iFPUA==",
"dev": true,
"dependencies": {
"@ampproject/remapping": "2.2.1",
- "@angular-devkit/architect": "0.1700.7",
- "@angular-devkit/build-webpack": "0.1700.7",
- "@angular-devkit/core": "17.0.7",
+ "@angular-devkit/architect": "0.1700.8",
+ "@angular-devkit/build-webpack": "0.1700.8",
+ "@angular-devkit/core": "17.0.8",
"@babel/core": "7.23.2",
"@babel/generator": "7.23.0",
"@babel/helper-annotate-as-pure": "7.22.5",
@@ -141,7 +141,7 @@
"@babel/preset-env": "7.23.2",
"@babel/runtime": "7.23.2",
"@discoveryjs/json-ext": "0.5.7",
- "@ngtools/webpack": "17.0.7",
+ "@ngtools/webpack": "17.0.8",
"@vitejs/plugin-basic-ssl": "1.0.1",
"ansi-colors": "4.1.3",
"autoprefixer": "10.4.16",
@@ -245,12 +245,12 @@
}
},
"node_modules/@angular-devkit/build-webpack": {
- "version": "0.1700.7",
- "resolved": "https://registry.npmjs.org/@angular-devkit/build-webpack/-/build-webpack-0.1700.7.tgz",
- "integrity": "sha512-B9Mg/qYDpE5my8PJ3VPQyRSUV0Oq1bFUzU8s0ZpqEZl1URKc04pm0LtLmebrMIcUZgDiGk0RHaD+O1E9IV/bdQ==",
+ "version": "0.1700.8",
+ "resolved": "https://registry.npmjs.org/@angular-devkit/build-webpack/-/build-webpack-0.1700.8.tgz",
+ "integrity": "sha512-GA7QlCAlYB3uBkRaUYgIC/Vfajb9jMmouwYiAAEm34ZyP3ThFjdqsYd/A/exnuESt5o6Bh++C/PI34sV3lawRA==",
"dev": true,
"dependencies": {
- "@angular-devkit/architect": "0.1700.7",
+ "@angular-devkit/architect": "0.1700.8",
"rxjs": "7.8.1"
},
"engines": {
@@ -264,9 +264,9 @@
}
},
"node_modules/@angular-devkit/core": {
- "version": "17.0.7",
- "resolved": "https://registry.npmjs.org/@angular-devkit/core/-/core-17.0.7.tgz",
- "integrity": "sha512-vATobHo5O5tJba424hJfQWLb40GzvZPNsI74dcgSUTgrDph8ksmk5xB9OvEvf0INorQZ2IMphj/VIWj4/+JqSA==",
+ "version": "17.0.8",
+ "resolved": "https://registry.npmjs.org/@angular-devkit/core/-/core-17.0.8.tgz",
+ "integrity": "sha512-gI8+SOwGUwr0WOlFrhLjohLolMzcguuoR0LTZEcGjdXvQyPgH4NDSRIIrfWCdu+ZVhfy76o3zQYdYc9QN8NrjQ==",
"dev": true,
"dependencies": {
"ajv": "8.12.0",
@@ -291,12 +291,12 @@
}
},
"node_modules/@angular-devkit/schematics": {
- "version": "17.0.7",
- "resolved": "https://registry.npmjs.org/@angular-devkit/schematics/-/schematics-17.0.7.tgz",
- "integrity": "sha512-BY11OkJkM3xyXcvyD7x5kGY/c8Ufd4AfPvI0D9imhVxbns45Q48b1DlvCQvSnCJ/s+OwnkrYb/Efa70ZiaGu8A==",
+ "version": "17.0.8",
+ "resolved": "https://registry.npmjs.org/@angular-devkit/schematics/-/schematics-17.0.8.tgz",
+ "integrity": "sha512-syo814SVWfJvne448IijjZvpWbuqJsEutdNqHWLTewTfX2U3KrIAr/XRVcXQMuyMvLCDiuxjMgEJxOIP7mcIPw==",
"dev": true,
"dependencies": {
- "@angular-devkit/core": "17.0.7",
+ "@angular-devkit/core": "17.0.8",
"jsonc-parser": "3.2.0",
"magic-string": "0.30.5",
"ora": "5.4.1",
@@ -423,15 +423,15 @@
}
},
"node_modules/@angular/cli": {
- "version": "17.0.7",
- "resolved": "https://registry.npmjs.org/@angular/cli/-/cli-17.0.7.tgz",
- "integrity": "sha512-oSa0GVAQNA7wFbLJYeaO3kV4iUcbKEqXDLxcIE8s1GfHddBOlXH2P1T4fXonCBl5qvV+joP0G0+fs7I0w2utZQ==",
+ "version": "17.0.8",
+ "resolved": "https://registry.npmjs.org/@angular/cli/-/cli-17.0.8.tgz",
+ "integrity": "sha512-yZXYNLAFv9u2qypsVqtS+rRCsnjsIPYXr6TcI/r5buzOtC7UQ2lleYsWJqX47SsyGMk/o3gaYg5Bj2I5mmRDLA==",
"dev": true,
"dependencies": {
- "@angular-devkit/architect": "0.1700.7",
- "@angular-devkit/core": "17.0.7",
- "@angular-devkit/schematics": "17.0.7",
- "@schematics/angular": "17.0.7",
+ "@angular-devkit/architect": "0.1700.8",
+ "@angular-devkit/core": "17.0.8",
+ "@angular-devkit/schematics": "17.0.8",
+ "@schematics/angular": "17.0.8",
"@yarnpkg/lockfile": "1.1.0",
"ansi-colors": "4.1.3",
"ini": "4.1.1",
@@ -457,9 +457,9 @@
}
},
"node_modules/@angular/common": {
- "version": "17.0.7",
- "resolved": "https://registry.npmjs.org/@angular/common/-/common-17.0.7.tgz",
- "integrity": "sha512-bPPL6x0KOAOTxKSE2j4EWmEUOnqZYzOYiHzroa5b9UEyA9NvGkd9bm3zIxw8xcndRj1Ehcmvpi6KBLcYBBbWfg==",
+ "version": "17.0.8",
+ "resolved": "https://registry.npmjs.org/@angular/common/-/common-17.0.8.tgz",
+ "integrity": "sha512-fFfwtdg7H+OkqnvV/ENu8F8KGfgIiH16DDbQqYY5KQyyQB+SMsoVW29F1fGx6Y30s7ZlsLOy6cHhgrw74itkSw==",
"dependencies": {
"tslib": "^2.3.0"
},
@@ -467,14 +467,14 @@
"node": "^18.13.0 || >=20.9.0"
},
"peerDependencies": {
- "@angular/core": "17.0.7",
+ "@angular/core": "17.0.8",
"rxjs": "^6.5.3 || ^7.4.0"
}
},
"node_modules/@angular/compiler": {
- "version": "17.0.7",
- "resolved": "https://registry.npmjs.org/@angular/compiler/-/compiler-17.0.7.tgz",
- "integrity": "sha512-QHPuLti2c2tGZmOGZ0cfCHo4LxiHUkC27I0aZFDyQSSQqEI5obQGVlEREHysw0nsS3sYIcLvqcwcKcRtXlXtxQ==",
+ "version": "17.0.8",
+ "resolved": "https://registry.npmjs.org/@angular/compiler/-/compiler-17.0.8.tgz",
+ "integrity": "sha512-48jWypuhBGTrUUbkz1vB9gjbKKZ3hpuJ2DUUncd331Yw4tqkqZQbBa/E3ei4IHiCxEvW2uX3lI4AwlhuozmUtA==",
"dependencies": {
"tslib": "^2.3.0"
},
@@ -482,7 +482,7 @@
"node": "^18.13.0 || >=20.9.0"
},
"peerDependencies": {
- "@angular/core": "17.0.7"
+ "@angular/core": "17.0.8"
},
"peerDependenciesMeta": {
"@angular/core": {
@@ -491,9 +491,9 @@
}
},
"node_modules/@angular/compiler-cli": {
- "version": "17.0.7",
- "resolved": "https://registry.npmjs.org/@angular/compiler-cli/-/compiler-cli-17.0.7.tgz",
- "integrity": "sha512-YnL38idjIYtl3BXYpv+sVJKWGbUjHT6eyQSQVAfO/1AwWqVa21K9hnE+Q37VmUKEcKFMnQembeuErA+KVsGI6A==",
+ "version": "17.0.8",
+ "resolved": "https://registry.npmjs.org/@angular/compiler-cli/-/compiler-cli-17.0.8.tgz",
+ "integrity": "sha512-ny2SMVgl+icjMuU5ZM57yFGUrhjR0hNxfCn0otAD3jUFliz/Onu9l6EPRKA5Cr8MZx3mg3rTLSBMD17YT8rsOg==",
"dependencies": {
"@babel/core": "7.23.2",
"@jridgewell/sourcemap-codec": "^1.4.14",
@@ -513,14 +513,14 @@
"node": "^18.13.0 || >=20.9.0"
},
"peerDependencies": {
- "@angular/compiler": "17.0.7",
+ "@angular/compiler": "17.0.8",
"typescript": ">=5.2 <5.3"
}
},
"node_modules/@angular/core": {
- "version": "17.0.7",
- "resolved": "https://registry.npmjs.org/@angular/core/-/core-17.0.7.tgz",
- "integrity": "sha512-mEkelXkzEi6+A9GjdKOSGGzQAfo1iAjVTn6YsplNUeGE5JgDZYZ7sXGQqs0Lin7dzJxnPAgGjCOl7SpWLXIPSQ==",
+ "version": "17.0.8",
+ "resolved": "https://registry.npmjs.org/@angular/core/-/core-17.0.8.tgz",
+ "integrity": "sha512-tzYsK24LdkNuKNJK6efF4XOqspvF/qOe9j/n1Y61a6mNvFwsJFGbcmdZMby4hI/YRm6oIDoIIFjSep8ycp6Pbw==",
"dependencies": {
"tslib": "^2.3.0"
},
@@ -533,9 +533,9 @@
}
},
"node_modules/@angular/forms": {
- "version": "17.0.7",
- "resolved": "https://registry.npmjs.org/@angular/forms/-/forms-17.0.7.tgz",
- "integrity": "sha512-28BxRxEmgZIofGwVp6s2v3ri/kuWW+/EY/ZXhavlWKJEh4ATJl72k0RkRWNcQi4wnvn0Qb8tFdnVJnvRZvvKEw==",
+ "version": "17.0.8",
+ "resolved": "https://registry.npmjs.org/@angular/forms/-/forms-17.0.8.tgz",
+ "integrity": "sha512-WZBHbMQjaSovAzOMhKqZN+m7eUPGfOzh9rKFKvj6UQLIJ9qSpEpqlvL0omU1z/47s3XXeLiBzomMiRfQISJvvw==",
"dependencies": {
"tslib": "^2.3.0"
},
@@ -543,16 +543,16 @@
"node": "^18.13.0 || >=20.9.0"
},
"peerDependencies": {
- "@angular/common": "17.0.7",
- "@angular/core": "17.0.7",
- "@angular/platform-browser": "17.0.7",
+ "@angular/common": "17.0.8",
+ "@angular/core": "17.0.8",
+ "@angular/platform-browser": "17.0.8",
"rxjs": "^6.5.3 || ^7.4.0"
}
},
"node_modules/@angular/localize": {
- "version": "17.0.7",
- "resolved": "https://registry.npmjs.org/@angular/localize/-/localize-17.0.7.tgz",
- "integrity": "sha512-avYYQ8zin2thzvsH2YP3WxlwkvOzjNEXxjv4yyZBx6wul68e/753kQK/0RmSUYaBpDTUEZYzrPpDay00TKwBOA==",
+ "version": "17.0.8",
+ "resolved": "https://registry.npmjs.org/@angular/localize/-/localize-17.0.8.tgz",
+ "integrity": "sha512-1zW8qWKNMH3r/x4KpwzzUmVY+iN76vYdhjA6gzZDnpJxpon9eyljNEildj9+zSWeNUr2LgJ6HnkIX9q1f3mXfA==",
"dependencies": {
"@babel/core": "7.23.2",
"fast-glob": "3.3.1",
@@ -567,14 +567,14 @@
"node": "^18.13.0 || >=20.9.0"
},
"peerDependencies": {
- "@angular/compiler": "17.0.7",
- "@angular/compiler-cli": "17.0.7"
+ "@angular/compiler": "17.0.8",
+ "@angular/compiler-cli": "17.0.8"
}
},
"node_modules/@angular/platform-browser": {
- "version": "17.0.7",
- "resolved": "https://registry.npmjs.org/@angular/platform-browser/-/platform-browser-17.0.7.tgz",
- "integrity": "sha512-bm9/wt51nc/MPjft/FlRNIgFSeLjDtfJOT7M32Rt6kOHhNKSK7ZTPWdMe9ahuHSbAhLzd0G/4NsT5sKrWSeVZg==",
+ "version": "17.0.8",
+ "resolved": "https://registry.npmjs.org/@angular/platform-browser/-/platform-browser-17.0.8.tgz",
+ "integrity": "sha512-XaI+p2AxQaIHzR761lhPUf4OcOp46WDW0IfbvOzaezHE+8r81joZyVSDQPgXSa/aRfI58YhcfUavuGqyU3PphA==",
"dependencies": {
"tslib": "^2.3.0"
},
@@ -582,9 +582,9 @@
"node": "^18.13.0 || >=20.9.0"
},
"peerDependencies": {
- "@angular/animations": "17.0.7",
- "@angular/common": "17.0.7",
- "@angular/core": "17.0.7"
+ "@angular/animations": "17.0.8",
+ "@angular/common": "17.0.8",
+ "@angular/core": "17.0.8"
},
"peerDependenciesMeta": {
"@angular/animations": {
@@ -593,9 +593,9 @@
}
},
"node_modules/@angular/platform-browser-dynamic": {
- "version": "17.0.7",
- "resolved": "https://registry.npmjs.org/@angular/platform-browser-dynamic/-/platform-browser-dynamic-17.0.7.tgz",
- "integrity": "sha512-OquwUX9fLWA2JUZW5Jm6atk0CPt0sA7Tg24eGLsr6g1XfTS7jRZprlGaa72NgPLnQVV6m84o/ZiNYS6yPmq1Gg==",
+ "version": "17.0.8",
+ "resolved": "https://registry.npmjs.org/@angular/platform-browser-dynamic/-/platform-browser-dynamic-17.0.8.tgz",
+ "integrity": "sha512-BIXNKnfBZb8sdluQ7WIhIXFuVnsJJ0SV+aiMKzQ7B6XhWoAXZQnlvON2thydjIIVuCvaF3YmWTbILI2K8YZ2jQ==",
"dependencies": {
"tslib": "^2.3.0"
},
@@ -603,16 +603,16 @@
"node": "^18.13.0 || >=20.9.0"
},
"peerDependencies": {
- "@angular/common": "17.0.7",
- "@angular/compiler": "17.0.7",
- "@angular/core": "17.0.7",
- "@angular/platform-browser": "17.0.7"
+ "@angular/common": "17.0.8",
+ "@angular/compiler": "17.0.8",
+ "@angular/core": "17.0.8",
+ "@angular/platform-browser": "17.0.8"
}
},
"node_modules/@angular/router": {
- "version": "17.0.7",
- "resolved": "https://registry.npmjs.org/@angular/router/-/router-17.0.7.tgz",
- "integrity": "sha512-rUFPe1uDlYYw6+3Gq68czW7WxBH7zT/D3UsT1otqwUV4RnQQsVze4fIit9FqJh7tuP4y3WpB4XBNf7p7Oi6TJw==",
+ "version": "17.0.8",
+ "resolved": "https://registry.npmjs.org/@angular/router/-/router-17.0.8.tgz",
+ "integrity": "sha512-ptphcRe1RG/mIS60R7ZPilkkrxautqB0sOhds3h5VP3g628G1a2HWzvnmvjEfpJWDMFivV32VJMMBtTLqGr+0Q==",
"dependencies": {
"tslib": "^2.3.0"
},
@@ -620,9 +620,9 @@
"node": "^18.13.0 || >=20.9.0"
},
"peerDependencies": {
- "@angular/common": "17.0.7",
- "@angular/core": "17.0.7",
- "@angular/platform-browser": "17.0.7",
+ "@angular/common": "17.0.8",
+ "@angular/core": "17.0.8",
+ "@angular/platform-browser": "17.0.8",
"rxjs": "^6.5.3 || ^7.4.0"
}
},
@@ -2895,9 +2895,9 @@
}
},
"node_modules/@eslint/js": {
- "version": "8.55.0",
- "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.55.0.tgz",
- "integrity": "sha512-qQfo2mxH5yVom1kacMtZZJFVdW+E70mqHMJvVg6WTLo+VBuQJ4TojZlfWBjK0ve5BdEeNAVxOsl/nvNMpJOaJA==",
+ "version": "8.56.0",
+ "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.56.0.tgz",
+ "integrity": "sha512-gMsVel9D7f2HLkBma9VbtzZRehRogVRfbr++f06nL2vnCGCNlzOD+/MUov/F4p8myyAHspEhVobgjpX64q5m6A==",
"dev": true,
"engines": {
"node": "^12.22.0 || ^14.17.0 || >=16.0.0"
@@ -3941,9 +3941,9 @@
}
},
"node_modules/@ngtools/webpack": {
- "version": "17.0.7",
- "resolved": "https://registry.npmjs.org/@ngtools/webpack/-/webpack-17.0.7.tgz",
- "integrity": "sha512-gwhUhpwXn0trwwKdSu9WlJbEcLt+s/2fPwoD9lZ0y3wXfrOogsfcNBJKeO5BZf1h+A3AWt7ePmgrZXSJM+865Q==",
+ "version": "17.0.8",
+ "resolved": "https://registry.npmjs.org/@ngtools/webpack/-/webpack-17.0.8.tgz",
+ "integrity": "sha512-wx0XBMrbpDeailK2uIhp/ZVMC3GK3BWwJjUu5SbT4BFrcoi2Zd9/9m0RCBAY54UXLBCqKd+ih7pJ6JSvprZmWw==",
"dev": true,
"engines": {
"node": "^18.13.0 || >=20.9.0",
@@ -4459,13 +4459,13 @@
}
},
"node_modules/@schematics/angular": {
- "version": "17.0.7",
- "resolved": "https://registry.npmjs.org/@schematics/angular/-/angular-17.0.7.tgz",
- "integrity": "sha512-d7QKmcKrM4owb/2bR7Ipf23roiNbvbD/x7reNhQAtKAPLSHJ3Ulkf1+Yv+dj+9f+K7y9SBviEUSrD27BQ9WaxQ==",
+ "version": "17.0.8",
+ "resolved": "https://registry.npmjs.org/@schematics/angular/-/angular-17.0.8.tgz",
+ "integrity": "sha512-1h5mwKFv1B/L5JWZ0mxnC4ms06iwnSi/w+GgRZPeM3P5BpuZuvAkFiClNnM55iLlQJXRQioPNLM3sOsz7spR6w==",
"dev": true,
"dependencies": {
- "@angular-devkit/core": "17.0.7",
- "@angular-devkit/schematics": "17.0.7",
+ "@angular-devkit/core": "17.0.8",
+ "@angular-devkit/schematics": "17.0.8",
"jsonc-parser": "3.2.0"
},
"engines": {
@@ -4878,9 +4878,9 @@
"dev": true
},
"node_modules/@types/node": {
- "version": "20.10.4",
- "resolved": "https://registry.npmjs.org/@types/node/-/node-20.10.4.tgz",
- "integrity": "sha512-D08YG6rr8X90YB56tSIuBaddy/UXAA9RKJoFvrsnogAum/0pmjkgi4+2nx96A330FmioegBWmEYQ+syqCFaveg==",
+ "version": "20.10.6",
+ "resolved": "https://registry.npmjs.org/@types/node/-/node-20.10.6.tgz",
+ "integrity": "sha512-Vac8H+NlRNNlAmDfGUP7b5h/KA+AtWIzuXy0E6OyP8f1tCLYAtPvKRRDJjAPqhpCb0t6U2j7/xqAuLEebW2kiw==",
"dev": true,
"dependencies": {
"undici-types": "~5.26.4"
@@ -4896,9 +4896,9 @@
}
},
"node_modules/@types/qs": {
- "version": "6.9.10",
- "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.10.tgz",
- "integrity": "sha512-3Gnx08Ns1sEoCrWssEgTSJs/rsT2vhGP+Ja9cnnk9k4ALxinORlQneLXFeFKOTJMOeZUFD1s7w+w2AphTpvzZw==",
+ "version": "6.9.11",
+ "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.11.tgz",
+ "integrity": "sha512-oGk0gmhnEJK4Yyk+oI7EfXsLayXatCWPHary1MtcmbAifkobT9cM9yutG/hZKIseOU0MqbIwQ/u2nn/Gb+ltuQ==",
"dev": true
},
"node_modules/@types/range-parser": {
@@ -4995,16 +4995,16 @@
"dev": true
},
"node_modules/@typescript-eslint/eslint-plugin": {
- "version": "6.14.0",
- "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-6.14.0.tgz",
- "integrity": "sha512-1ZJBykBCXaSHG94vMMKmiHoL0MhNHKSVlcHVYZNw+BKxufhqQVTOawNpwwI1P5nIFZ/4jLVop0mcY6mJJDFNaw==",
+ "version": "6.17.0",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-6.17.0.tgz",
+ "integrity": "sha512-Vih/4xLXmY7V490dGwBQJTpIZxH4ZFH6eCVmQ4RFkB+wmaCTDAx4dtgoWwMNGKLkqRY1L6rPqzEbjorRnDo4rQ==",
"dev": true,
"dependencies": {
"@eslint-community/regexpp": "^4.5.1",
- "@typescript-eslint/scope-manager": "6.14.0",
- "@typescript-eslint/type-utils": "6.14.0",
- "@typescript-eslint/utils": "6.14.0",
- "@typescript-eslint/visitor-keys": "6.14.0",
+ "@typescript-eslint/scope-manager": "6.17.0",
+ "@typescript-eslint/type-utils": "6.17.0",
+ "@typescript-eslint/utils": "6.17.0",
+ "@typescript-eslint/visitor-keys": "6.17.0",
"debug": "^4.3.4",
"graphemer": "^1.4.0",
"ignore": "^5.2.4",
@@ -5030,13 +5030,13 @@
}
},
"node_modules/@typescript-eslint/eslint-plugin/node_modules/@typescript-eslint/type-utils": {
- "version": "6.14.0",
- "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-6.14.0.tgz",
- "integrity": "sha512-x6OC9Q7HfYKqjnuNu5a7kffIYs3No30isapRBJl1iCHLitD8O0lFbRcVGiOcuyN837fqXzPZ1NS10maQzZMKqw==",
+ "version": "6.17.0",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-6.17.0.tgz",
+ "integrity": "sha512-hDXcWmnbtn4P2B37ka3nil3yi3VCQO2QEB9gBiHJmQp5wmyQWqnjA85+ZcE8c4FqnaB6lBwMrPkgd4aBYz3iNg==",
"dev": true,
"dependencies": {
- "@typescript-eslint/typescript-estree": "6.14.0",
- "@typescript-eslint/utils": "6.14.0",
+ "@typescript-eslint/typescript-estree": "6.17.0",
+ "@typescript-eslint/utils": "6.17.0",
"debug": "^4.3.4",
"ts-api-utils": "^1.0.1"
},
@@ -5057,17 +5057,17 @@
}
},
"node_modules/@typescript-eslint/eslint-plugin/node_modules/@typescript-eslint/utils": {
- "version": "6.14.0",
- "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-6.14.0.tgz",
- "integrity": "sha512-XwRTnbvRr7Ey9a1NT6jqdKX8y/atWG+8fAIu3z73HSP8h06i3r/ClMhmaF/RGWGW1tHJEwij1uEg2GbEmPYvYg==",
+ "version": "6.17.0",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-6.17.0.tgz",
+ "integrity": "sha512-LofsSPjN/ITNkzV47hxas2JCsNCEnGhVvocfyOcLzT9c/tSZE7SfhS/iWtzP1lKNOEfLhRTZz6xqI8N2RzweSQ==",
"dev": true,
"dependencies": {
"@eslint-community/eslint-utils": "^4.4.0",
"@types/json-schema": "^7.0.12",
"@types/semver": "^7.5.0",
- "@typescript-eslint/scope-manager": "6.14.0",
- "@typescript-eslint/types": "6.14.0",
- "@typescript-eslint/typescript-estree": "6.14.0",
+ "@typescript-eslint/scope-manager": "6.17.0",
+ "@typescript-eslint/types": "6.17.0",
+ "@typescript-eslint/typescript-estree": "6.17.0",
"semver": "^7.5.4"
},
"engines": {
@@ -5082,15 +5082,15 @@
}
},
"node_modules/@typescript-eslint/parser": {
- "version": "6.14.0",
- "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-6.14.0.tgz",
- "integrity": "sha512-QjToC14CKacd4Pa7JK4GeB/vHmWFJckec49FR4hmIRf97+KXole0T97xxu9IFiPxVQ1DBWrQ5wreLwAGwWAVQA==",
+ "version": "6.17.0",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-6.17.0.tgz",
+ "integrity": "sha512-C4bBaX2orvhK+LlwrY8oWGmSl4WolCfYm513gEccdWZj0CwGadbIADb0FtVEcI+WzUyjyoBj2JRP8g25E6IB8A==",
"dev": true,
"dependencies": {
- "@typescript-eslint/scope-manager": "6.14.0",
- "@typescript-eslint/types": "6.14.0",
- "@typescript-eslint/typescript-estree": "6.14.0",
- "@typescript-eslint/visitor-keys": "6.14.0",
+ "@typescript-eslint/scope-manager": "6.17.0",
+ "@typescript-eslint/types": "6.17.0",
+ "@typescript-eslint/typescript-estree": "6.17.0",
+ "@typescript-eslint/visitor-keys": "6.17.0",
"debug": "^4.3.4"
},
"engines": {
@@ -5110,13 +5110,13 @@
}
},
"node_modules/@typescript-eslint/scope-manager": {
- "version": "6.14.0",
- "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-6.14.0.tgz",
- "integrity": "sha512-VT7CFWHbZipPncAZtuALr9y3EuzY1b1t1AEkIq2bTXUPKw+pHoXflGNG5L+Gv6nKul1cz1VH8fz16IThIU0tdg==",
+ "version": "6.17.0",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-6.17.0.tgz",
+ "integrity": "sha512-RX7a8lwgOi7am0k17NUO0+ZmMOX4PpjLtLRgLmT1d3lBYdWH4ssBUbwdmc5pdRX8rXon8v9x8vaoOSpkHfcXGA==",
"dev": true,
"dependencies": {
- "@typescript-eslint/types": "6.14.0",
- "@typescript-eslint/visitor-keys": "6.14.0"
+ "@typescript-eslint/types": "6.17.0",
+ "@typescript-eslint/visitor-keys": "6.17.0"
},
"engines": {
"node": "^16.0.0 || >=18.0.0"
@@ -5211,9 +5211,9 @@
}
},
"node_modules/@typescript-eslint/types": {
- "version": "6.14.0",
- "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-6.14.0.tgz",
- "integrity": "sha512-uty9H2K4Xs8E47z3SnXEPRNDfsis8JO27amp2GNCnzGETEW3yTqEIVg5+AI7U276oGF/tw6ZA+UesxeQ104ceA==",
+ "version": "6.17.0",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-6.17.0.tgz",
+ "integrity": "sha512-qRKs9tvc3a4RBcL/9PXtKSehI/q8wuU9xYJxe97WFxnzH8NWWtcW3ffNS+EWg8uPvIerhjsEZ+rHtDqOCiH57A==",
"dev": true,
"engines": {
"node": "^16.0.0 || >=18.0.0"
@@ -5224,16 +5224,17 @@
}
},
"node_modules/@typescript-eslint/typescript-estree": {
- "version": "6.14.0",
- "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-6.14.0.tgz",
- "integrity": "sha512-yPkaLwK0yH2mZKFE/bXkPAkkFgOv15GJAUzgUVonAbv0Hr4PK/N2yaA/4XQbTZQdygiDkpt5DkxPELqHguNvyw==",
+ "version": "6.17.0",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-6.17.0.tgz",
+ "integrity": "sha512-gVQe+SLdNPfjlJn5VNGhlOhrXz4cajwFd5kAgWtZ9dCZf4XJf8xmgCTLIqec7aha3JwgLI2CK6GY1043FRxZwg==",
"dev": true,
"dependencies": {
- "@typescript-eslint/types": "6.14.0",
- "@typescript-eslint/visitor-keys": "6.14.0",
+ "@typescript-eslint/types": "6.17.0",
+ "@typescript-eslint/visitor-keys": "6.17.0",
"debug": "^4.3.4",
"globby": "^11.1.0",
"is-glob": "^4.0.3",
+ "minimatch": "9.0.3",
"semver": "^7.5.4",
"ts-api-utils": "^1.0.1"
},
@@ -5250,6 +5251,30 @@
}
}
},
+ "node_modules/@typescript-eslint/typescript-estree/node_modules/brace-expansion": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz",
+ "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==",
+ "dev": true,
+ "dependencies": {
+ "balanced-match": "^1.0.0"
+ }
+ },
+ "node_modules/@typescript-eslint/typescript-estree/node_modules/minimatch": {
+ "version": "9.0.3",
+ "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.3.tgz",
+ "integrity": "sha512-RHiac9mvaRw0x3AYRgDC1CxAP7HTcNrrECeA8YYJeWnpo+2Q5CegtZjaotWTWxDG3UeGA1coE05iH1mPjT/2mg==",
+ "dev": true,
+ "dependencies": {
+ "brace-expansion": "^2.0.1"
+ },
+ "engines": {
+ "node": ">=16 || 14 >=14.17"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/isaacs"
+ }
+ },
"node_modules/@typescript-eslint/utils": {
"version": "6.13.1",
"resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-6.13.1.tgz",
@@ -5350,12 +5375,12 @@
}
},
"node_modules/@typescript-eslint/visitor-keys": {
- "version": "6.14.0",
- "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-6.14.0.tgz",
- "integrity": "sha512-fB5cw6GRhJUz03MrROVuj5Zm/Q+XWlVdIsFj+Zb1Hvqouc8t+XP2H5y53QYU/MGtd2dPg6/vJJlhoX3xc2ehfw==",
+ "version": "6.17.0",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-6.17.0.tgz",
+ "integrity": "sha512-H6VwB/k3IuIeQOyYczyyKN8wH6ed8EwliaYHLxOIhyF0dYEIsN8+Bk3GE19qafeMKyZJJHP8+O1HiFhFLUNKSg==",
"dev": true,
"dependencies": {
- "@typescript-eslint/types": "6.14.0",
+ "@typescript-eslint/types": "6.17.0",
"eslint-visitor-keys": "^3.4.1"
},
"engines": {
@@ -8687,15 +8712,15 @@
}
},
"node_modules/eslint": {
- "version": "8.55.0",
- "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.55.0.tgz",
- "integrity": "sha512-iyUUAM0PCKj5QpwGfmCAG9XXbZCWsqP/eWAWrG/W0umvjuLRBECwSFdt+rCntju0xEH7teIABPwXpahftIaTdA==",
+ "version": "8.56.0",
+ "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.56.0.tgz",
+ "integrity": "sha512-Go19xM6T9puCOWntie1/P997aXxFsOi37JIHRWI514Hc6ZnaHGKY9xFhrU65RT6CcBEzZoGG1e6Nq+DT04ZtZQ==",
"dev": true,
"dependencies": {
"@eslint-community/eslint-utils": "^4.2.0",
"@eslint-community/regexpp": "^4.6.1",
"@eslint/eslintrc": "^2.1.4",
- "@eslint/js": "8.55.0",
+ "@eslint/js": "8.56.0",
"@humanwhocodes/config-array": "^0.11.13",
"@humanwhocodes/module-importer": "^1.0.1",
"@nodelib/fs.walk": "^1.2.8",
diff --git a/src-ui/package.json b/src-ui/package.json
index fddb9677c..904c2b9a0 100644
--- a/src-ui/package.json
+++ b/src-ui/package.json
@@ -12,14 +12,14 @@
"private": true,
"dependencies": {
"@angular/cdk": "^17.0.4",
- "@angular/common": "~17.0.7",
- "@angular/compiler": "~17.0.7",
- "@angular/core": "~17.0.7",
- "@angular/forms": "~17.0.7",
- "@angular/localize": "~17.0.7",
- "@angular/platform-browser": "~17.0.7",
- "@angular/platform-browser-dynamic": "~17.0.7",
- "@angular/router": "~17.0.7",
+ "@angular/common": "~17.0.8",
+ "@angular/compiler": "~17.0.8",
+ "@angular/core": "~17.0.8",
+ "@angular/forms": "~17.0.8",
+ "@angular/localize": "~17.0.8",
+ "@angular/platform-browser": "~17.0.8",
+ "@angular/platform-browser-dynamic": "~17.0.8",
+ "@angular/router": "~17.0.8",
"@ng-bootstrap/ng-bootstrap": "^16.0.0",
"@ng-select/ng-select": "^12.0.4",
"@ngneat/dirty-check-forms": "^3.0.3",
@@ -39,21 +39,21 @@
},
"devDependencies": {
"@angular-builders/jest": "17.0.0",
- "@angular-devkit/build-angular": "~17.0.7",
+ "@angular-devkit/build-angular": "~17.0.8",
"@angular-eslint/builder": "17.1.1",
"@angular-eslint/eslint-plugin": "17.1.1",
"@angular-eslint/eslint-plugin-template": "17.1.1",
"@angular-eslint/schematics": "17.1.1",
"@angular-eslint/template-parser": "17.1.1",
- "@angular/cli": "~17.0.7",
+ "@angular/cli": "~17.0.8",
"@angular/compiler-cli": "~17.0.7",
"@playwright/test": "^1.40.1",
"@types/jest": "^29.5.10",
- "@types/node": "^20.10.2",
- "@typescript-eslint/eslint-plugin": "^6.10.0",
- "@typescript-eslint/parser": "^6.10.0",
+ "@types/node": "^20.10.6",
+ "@typescript-eslint/eslint-plugin": "^6.17.0",
+ "@typescript-eslint/parser": "^6.17.0",
"concurrently": "^8.2.2",
- "eslint": "^8.53.0",
+ "eslint": "^8.56.0",
"jest": "29.7.0",
"jest-environment-jsdom": "^29.7.0",
"jest-preset-angular": "^13.1.4",
diff --git a/src-ui/src/app/app-routing.module.ts b/src-ui/src/app/app-routing.module.ts
index b3952634c..6da2cd253 100644
--- a/src-ui/src/app/app-routing.module.ts
+++ b/src-ui/src/app/app-routing.module.ts
@@ -21,10 +21,11 @@ import {
PermissionAction,
PermissionType,
} from './services/permissions.service'
-import { ConsumptionTemplatesComponent } from './components/manage/consumption-templates/consumption-templates.component'
+import { WorkflowsComponent } from './components/manage/workflows/workflows.component'
import { MailComponent } from './components/manage/mail/mail.component'
import { UsersAndGroupsComponent } from './components/admin/users-groups/users-groups.component'
import { CustomFieldsComponent } from './components/manage/custom-fields/custom-fields.component'
+import { ConfigComponent } from './components/admin/config/config.component'
export const routes: Routes = [
{ path: '', redirectTo: 'dashboard', pathMatch: 'full' },
@@ -179,6 +180,17 @@ export const routes: Routes = [
},
},
},
+ {
+ path: 'config',
+ component: ConfigComponent,
+ canActivate: [PermissionsGuard],
+ data: {
+ requiredPermission: {
+ action: PermissionAction.View,
+ type: PermissionType.Admin,
+ },
+ },
+ },
{
path: 'tasks',
component: TasksComponent,
@@ -202,13 +214,13 @@ export const routes: Routes = [
},
},
{
- path: 'templates',
- component: ConsumptionTemplatesComponent,
+ path: 'workflows',
+ component: WorkflowsComponent,
canActivate: [PermissionsGuard],
data: {
requiredPermission: {
action: PermissionAction.View,
- type: PermissionType.ConsumptionTemplate,
+ type: PermissionType.Workflow,
},
},
},
diff --git a/src-ui/src/app/app.component.ts b/src-ui/src/app/app.component.ts
index 2ab37d55e..e93fde30c 100644
--- a/src-ui/src/app/app.component.ts
+++ b/src-ui/src/app/app.component.ts
@@ -176,9 +176,9 @@ export class AppComponent implements OnInit, OnDestroy {
},
},
{
- anchorId: 'tour.consumption-templates',
- content: $localize`Consumption templates give you finer control over the document ingestion process.`,
- route: '/templates',
+ anchorId: 'tour.workflows',
+ content: $localize`Workflows give you more control over the document pipeline.`,
+ route: '/workflows',
backdropConfig: {
offset: 0,
},
diff --git a/src-ui/src/app/app.module.ts b/src-ui/src/app/app.module.ts
index c3b98549a..ad76bdb74 100644
--- a/src-ui/src/app/app.module.ts
+++ b/src-ui/src/app/app.module.ts
@@ -95,8 +95,8 @@ import { UsernamePipe } from './pipes/username.pipe'
import { LogoComponent } from './components/common/logo/logo.component'
import { IsNumberPipe } from './pipes/is-number.pipe'
import { ShareLinksDropdownComponent } from './components/common/share-links-dropdown/share-links-dropdown.component'
-import { ConsumptionTemplatesComponent } from './components/manage/consumption-templates/consumption-templates.component'
-import { ConsumptionTemplateEditDialogComponent } from './components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component'
+import { WorkflowsComponent } from './components/manage/workflows/workflows.component'
+import { WorkflowEditDialogComponent } from './components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component'
import { MailComponent } from './components/manage/mail/mail.component'
import { UsersAndGroupsComponent } from './components/admin/users-groups/users-groups.component'
import { DragDropModule } from '@angular/cdk/drag-drop'
@@ -108,6 +108,8 @@ import { ProfileEditDialogComponent } from './components/common/profile-edit-dia
import { PdfViewerComponent } from './components/common/pdf-viewer/pdf-viewer.component'
import { DocumentLinkComponent } from './components/common/input/document-link/document-link.component'
import { PreviewPopupComponent } from './components/common/preview-popup/preview-popup.component'
+import { SwitchComponent } from './components/common/input/switch/switch.component'
+import { ConfigComponent } from './components/admin/config/config.component'
import localeAf from '@angular/common/locales/af'
import localeAr from '@angular/common/locales/ar'
@@ -251,8 +253,8 @@ function initializeApp(settings: SettingsService) {
LogoComponent,
IsNumberPipe,
ShareLinksDropdownComponent,
- ConsumptionTemplatesComponent,
- ConsumptionTemplateEditDialogComponent,
+ WorkflowsComponent,
+ WorkflowEditDialogComponent,
MailComponent,
UsersAndGroupsComponent,
FileDropComponent,
@@ -263,6 +265,8 @@ function initializeApp(settings: SettingsService) {
PdfViewerComponent,
DocumentLinkComponent,
PreviewPopupComponent,
+ SwitchComponent,
+ ConfigComponent,
],
imports: [
BrowserModule,
diff --git a/src-ui/src/app/components/admin/config/config.component.html b/src-ui/src/app/components/admin/config/config.component.html
new file mode 100644
index 000000000..a3eb0b8ab
--- /dev/null
+++ b/src-ui/src/app/components/admin/config/config.component.html
@@ -0,0 +1,54 @@
+
+
+
diff --git a/src-ui/src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.scss b/src-ui/src/app/components/admin/config/config.component.scss
similarity index 100%
rename from src-ui/src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.scss
rename to src-ui/src/app/components/admin/config/config.component.scss
diff --git a/src-ui/src/app/components/admin/config/config.component.spec.ts b/src-ui/src/app/components/admin/config/config.component.spec.ts
new file mode 100644
index 000000000..5d70881b6
--- /dev/null
+++ b/src-ui/src/app/components/admin/config/config.component.spec.ts
@@ -0,0 +1,103 @@
+import { ComponentFixture, TestBed } from '@angular/core/testing'
+
+import { ConfigComponent } from './config.component'
+import { ConfigService } from 'src/app/services/config.service'
+import { ToastService } from 'src/app/services/toast.service'
+import { of, throwError } from 'rxjs'
+import { OutputTypeConfig } from 'src/app/data/paperless-config'
+import { HttpClientTestingModule } from '@angular/common/http/testing'
+import { BrowserModule } from '@angular/platform-browser'
+import { NgbModule } from '@ng-bootstrap/ng-bootstrap'
+import { NgSelectModule } from '@ng-select/ng-select'
+import { TextComponent } from '../../common/input/text/text.component'
+import { NumberComponent } from '../../common/input/number/number.component'
+import { SwitchComponent } from '../../common/input/switch/switch.component'
+import { FormsModule, ReactiveFormsModule } from '@angular/forms'
+import { PageHeaderComponent } from '../../common/page-header/page-header.component'
+import { SelectComponent } from '../../common/input/select/select.component'
+
+describe('ConfigComponent', () => {
+ let component: ConfigComponent
+ let fixture: ComponentFixture
+ let configService: ConfigService
+ let toastService: ToastService
+
+ beforeEach(async () => {
+ await TestBed.configureTestingModule({
+ declarations: [
+ ConfigComponent,
+ TextComponent,
+ SelectComponent,
+ NumberComponent,
+ SwitchComponent,
+ PageHeaderComponent,
+ ],
+ imports: [
+ HttpClientTestingModule,
+ BrowserModule,
+ NgbModule,
+ NgSelectModule,
+ FormsModule,
+ ReactiveFormsModule,
+ ],
+ }).compileComponents()
+
+ configService = TestBed.inject(ConfigService)
+ toastService = TestBed.inject(ToastService)
+ fixture = TestBed.createComponent(ConfigComponent)
+ component = fixture.componentInstance
+ fixture.detectChanges()
+ })
+
+ it('should load config on init, show error if necessary', () => {
+ const getSpy = jest.spyOn(configService, 'getConfig')
+ const errorSpy = jest.spyOn(toastService, 'showError')
+ getSpy.mockReturnValueOnce(
+ throwError(() => new Error('Error getting config'))
+ )
+ component.ngOnInit()
+ expect(getSpy).toHaveBeenCalled()
+ expect(errorSpy).toHaveBeenCalled()
+ getSpy.mockReturnValueOnce(
+ of({ output_type: OutputTypeConfig.PDF_A } as any)
+ )
+ component.ngOnInit()
+ expect(component.initialConfig).toEqual({
+ output_type: OutputTypeConfig.PDF_A,
+ })
+ })
+
+ it('should save config, show error if necessary', () => {
+ const saveSpy = jest.spyOn(configService, 'saveConfig')
+ const errorSpy = jest.spyOn(toastService, 'showError')
+ saveSpy.mockReturnValueOnce(
+ throwError(() => new Error('Error saving config'))
+ )
+ component.saveConfig()
+ expect(saveSpy).toHaveBeenCalled()
+ expect(errorSpy).toHaveBeenCalled()
+ saveSpy.mockReturnValueOnce(
+ of({ output_type: OutputTypeConfig.PDF_A } as any)
+ )
+ component.saveConfig()
+ expect(component.initialConfig).toEqual({
+ output_type: OutputTypeConfig.PDF_A,
+ })
+ })
+
+ it('should support discard changes', () => {
+ component.initialConfig = { output_type: OutputTypeConfig.PDF_A2 } as any
+ component.configForm.patchValue({ output_type: OutputTypeConfig.PDF_A })
+ component.discardChanges()
+ expect(component.configForm.get('output_type').value).toEqual(
+ OutputTypeConfig.PDF_A2
+ )
+ })
+
+ it('should support JSON validation for e.g. user_args', () => {
+ component.configForm.patchValue({ user_args: '{ foo bar }' })
+ expect(component.errors).toEqual({ user_args: 'Invalid JSON' })
+ component.configForm.patchValue({ user_args: '{ "foo": "bar" }' })
+ expect(component.errors).toEqual({ user_args: null })
+ })
+})
diff --git a/src-ui/src/app/components/admin/config/config.component.ts b/src-ui/src/app/components/admin/config/config.component.ts
new file mode 100644
index 000000000..66d7b537f
--- /dev/null
+++ b/src-ui/src/app/components/admin/config/config.component.ts
@@ -0,0 +1,163 @@
+import { Component, OnDestroy, OnInit } from '@angular/core'
+import { AbstractControl, FormControl, FormGroup } from '@angular/forms'
+import {
+ BehaviorSubject,
+ Observable,
+ Subject,
+ Subscription,
+ first,
+ takeUntil,
+} from 'rxjs'
+import {
+ PaperlessConfigOptions,
+ ConfigCategory,
+ ConfigOption,
+ ConfigOptionType,
+ PaperlessConfig,
+} from 'src/app/data/paperless-config'
+import { ConfigService } from 'src/app/services/config.service'
+import { ToastService } from 'src/app/services/toast.service'
+import { ComponentWithPermissions } from '../../with-permissions/with-permissions.component'
+import { DirtyComponent, dirtyCheck } from '@ngneat/dirty-check-forms'
+
+@Component({
+ selector: 'pngx-config',
+ templateUrl: './config.component.html',
+ styleUrl: './config.component.scss',
+})
+export class ConfigComponent
+ extends ComponentWithPermissions
+ implements OnInit, OnDestroy, DirtyComponent
+{
+ public readonly ConfigOptionType = ConfigOptionType
+
+ // generated dynamically
+ public configForm = new FormGroup({})
+
+ public errors = {}
+
+ get optionCategories(): string[] {
+ return Object.values(ConfigCategory)
+ }
+
+ getCategoryOptions(category: string): ConfigOption[] {
+ return PaperlessConfigOptions.filter((o) => o.category === category)
+ }
+
+ public loading: boolean = false
+
+ initialConfig: PaperlessConfig
+ store: BehaviorSubject
+ storeSub: Subscription
+ isDirty$: Observable
+
+ private unsubscribeNotifier: Subject = new Subject()
+
+ constructor(
+ private configService: ConfigService,
+ private toastService: ToastService
+ ) {
+ super()
+ this.configForm.addControl('id', new FormControl())
+ PaperlessConfigOptions.forEach((option) => {
+ this.configForm.addControl(option.key, new FormControl())
+ })
+ }
+
+ ngOnInit(): void {
+ this.loading = true
+ this.configService
+ .getConfig()
+ .pipe(takeUntil(this.unsubscribeNotifier))
+ .subscribe({
+ next: (config) => {
+ this.loading = false
+ this.initialize(config)
+ },
+ error: (e) => {
+ this.loading = false
+ this.toastService.showError($localize`Error retrieving config`, e)
+ },
+ })
+
+ // validate JSON inputs
+ PaperlessConfigOptions.filter(
+ (o) => o.type === ConfigOptionType.JSON
+ ).forEach((option) => {
+ this.configForm
+ .get(option.key)
+ .addValidators((control: AbstractControl) => {
+ if (!control.value || control.value.toString().length === 0)
+ return null
+ try {
+ JSON.parse(control.value)
+ } catch (e) {
+ return [
+ {
+ user_args: e,
+ },
+ ]
+ }
+ return null
+ })
+ this.configForm.get(option.key).statusChanges.subscribe((status) => {
+ this.errors[option.key] =
+ status === 'INVALID' ? $localize`Invalid JSON` : null
+ })
+ this.configForm.get(option.key).updateValueAndValidity()
+ })
+ }
+
+ ngOnDestroy(): void {
+ this.unsubscribeNotifier.next(true)
+ this.unsubscribeNotifier.complete()
+ }
+
+ private initialize(config: PaperlessConfig) {
+ if (!this.store) {
+ this.store = new BehaviorSubject(config)
+
+ this.store
+ .asObservable()
+ .pipe(takeUntil(this.unsubscribeNotifier))
+ .subscribe((state) => {
+ this.configForm.patchValue(state, { emitEvent: false })
+ })
+
+ this.isDirty$ = dirtyCheck(this.configForm, this.store.asObservable())
+ }
+ this.configForm.patchValue(config)
+
+ this.initialConfig = config
+ }
+
+ getDocsUrl(key: string) {
+ return `https://docs.paperless-ngx.com/configuration/#${key}`
+ }
+
+ public saveConfig() {
+ this.loading = true
+ this.configService
+ .saveConfig(this.configForm.value as PaperlessConfig)
+ .pipe(takeUntil(this.unsubscribeNotifier), first())
+ .subscribe({
+ next: (config) => {
+ this.loading = false
+ this.initialize(config)
+ this.store.next(config)
+ this.toastService.showInfo($localize`Configuration updated`)
+ },
+ error: (e) => {
+ this.loading = false
+ this.toastService.showError(
+ $localize`An error occurred updating configuration`,
+ e
+ )
+ },
+ })
+ }
+
+ public discardChanges() {
+ this.configForm.reset(this.initialConfig)
+ }
+}
diff --git a/src-ui/src/app/components/app-frame/app-frame.component.html b/src-ui/src/app/components/app-frame/app-frame.component.html
index 2ab3fe0ae..32241333e 100644
--- a/src-ui/src/app/components/app-frame/app-frame.component.html
+++ b/src-ui/src/app/components/app-frame/app-frame.component.html
@@ -235,14 +235,14 @@