Compare commits
24 Commits
dev
...
feature-pa
Author | SHA1 | Date | |
---|---|---|---|
|
a0a9e0c6c8 | ||
|
1c7c703e5f | ||
|
53e9e910d8 | ||
|
9fe611a24c | ||
|
31e71aab83 | ||
|
7e7ce97d10 | ||
|
e06adc58c7 | ||
|
7170ac31b7 | ||
|
a0aa78c788 | ||
|
f3438914cc | ||
|
e1b944ce6b | ||
|
0add5aab0e | ||
|
c9adc74fa9 | ||
|
32abfbfc0a | ||
|
7f02f782f4 | ||
|
7c3f011e84 | ||
|
5c68177960 | ||
|
7a4666783e | ||
|
372825c271 | ||
|
abfddd6931 | ||
|
b3d49dbf12 | ||
|
673839265d | ||
|
f31df22ab6 | ||
|
f897447a65 |
@ -4099,6 +4099,18 @@
|
|||||||
<context context-type="sourcefile">src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html</context>
|
<context context-type="sourcefile">src/app/components/common/edit-dialog/workflow-edit-dialog/workflow-edit-dialog.component.html</context>
|
||||||
<context context-type="linenumber">111</context>
|
<context context-type="linenumber">111</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
|
<context-group purpose="location">
|
||||||
|
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
||||||
|
<context context-type="linenumber">165</context>
|
||||||
|
</context-group>
|
||||||
|
<context-group purpose="location">
|
||||||
|
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
||||||
|
<context context-type="linenumber">189</context>
|
||||||
|
</context-group>
|
||||||
|
<context-group purpose="location">
|
||||||
|
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
||||||
|
<context context-type="linenumber">213</context>
|
||||||
|
</context-group>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/common/toast/toast.component.html</context>
|
<context context-type="sourcefile">src/app/components/common/toast/toast.component.html</context>
|
||||||
<context context-type="linenumber">30</context>
|
<context context-type="linenumber">30</context>
|
||||||
@ -5548,7 +5560,7 @@
|
|||||||
</context-group>
|
</context-group>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
||||||
<context context-type="linenumber">156</context>
|
<context context-type="linenumber">231</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/manage/mail/mail.component.html</context>
|
<context context-type="sourcefile">src/app/components/manage/mail/mail.component.html</context>
|
||||||
@ -5943,77 +5955,98 @@
|
|||||||
<source>Migration Status</source>
|
<source>Migration Status</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
||||||
<context context-type="linenumber">56</context>
|
<context context-type="linenumber">65</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="7489316373554112115" datatype="html">
|
<trans-unit id="7489316373554112115" datatype="html">
|
||||||
<source>Up to date</source>
|
<source>Up to date</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
||||||
<context context-type="linenumber">59</context>
|
<context context-type="linenumber">69</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="7881311375431899727" datatype="html">
|
<trans-unit id="7881311375431899727" datatype="html">
|
||||||
<source>Latest Migration</source>
|
<source>Latest Migration</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
||||||
<context context-type="linenumber">64</context>
|
<context context-type="linenumber">74</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="4632965004151576238" datatype="html">
|
<trans-unit id="4632965004151576238" datatype="html">
|
||||||
<source>Pending Migrations</source>
|
<source>Pending Migrations</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
||||||
<context context-type="linenumber">66</context>
|
<context context-type="linenumber">76</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="6904866445262015585" datatype="html">
|
<trans-unit id="2790343143501919450" datatype="html">
|
||||||
<source>Tasks</source>
|
<source>Tasks Queue</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
||||||
<context context-type="linenumber">83</context>
|
<context context-type="linenumber">94</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="6911698235105017958" datatype="html">
|
<trans-unit id="6911698235105017958" datatype="html">
|
||||||
<source>Redis Status</source>
|
<source>Redis Status</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
||||||
<context context-type="linenumber">87</context>
|
<context context-type="linenumber">98</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="5349496739889768589" datatype="html">
|
<trans-unit id="5349496739889768589" datatype="html">
|
||||||
<source>Celery Status</source>
|
<source>Celery Status</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
||||||
<context context-type="linenumber">96</context>
|
<context context-type="linenumber">116</context>
|
||||||
|
</context-group>
|
||||||
|
</trans-unit>
|
||||||
|
<trans-unit id="2041675390931385838" datatype="html">
|
||||||
|
<source>Health</source>
|
||||||
|
<context-group purpose="location">
|
||||||
|
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
||||||
|
<context context-type="linenumber">142</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="31377277941774469" datatype="html">
|
<trans-unit id="31377277941774469" datatype="html">
|
||||||
<source>Search Index</source>
|
<source>Search Index</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
||||||
<context context-type="linenumber">105</context>
|
<context context-type="linenumber">146</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="4089509911694721896" datatype="html">
|
<trans-unit id="4089509911694721896" datatype="html">
|
||||||
<source>Last Updated</source>
|
<source>Last Updated</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
||||||
<context context-type="linenumber">119</context>
|
<context context-type="linenumber">163</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="46628344485199198" datatype="html">
|
<trans-unit id="46628344485199198" datatype="html">
|
||||||
<source>Classifier</source>
|
<source>Classifier</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
||||||
<context context-type="linenumber">121</context>
|
<context context-type="linenumber">168</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="6096684179126491743" datatype="html">
|
<trans-unit id="6096684179126491743" datatype="html">
|
||||||
<source>Last Trained</source>
|
<source>Last Trained</source>
|
||||||
<context-group purpose="location">
|
<context-group purpose="location">
|
||||||
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
||||||
<context context-type="linenumber">139</context>
|
<context context-type="linenumber">187</context>
|
||||||
|
</context-group>
|
||||||
|
</trans-unit>
|
||||||
|
<trans-unit id="6427836860962380759" datatype="html">
|
||||||
|
<source>Sanity Checker</source>
|
||||||
|
<context-group purpose="location">
|
||||||
|
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
||||||
|
<context context-type="linenumber">192</context>
|
||||||
|
</context-group>
|
||||||
|
</trans-unit>
|
||||||
|
<trans-unit id="6578747070254776938" datatype="html">
|
||||||
|
<source>Last Run</source>
|
||||||
|
<context-group purpose="location">
|
||||||
|
<context context-type="sourcefile">src/app/components/common/system-status-dialog/system-status-dialog.component.html</context>
|
||||||
|
<context context-type="linenumber">211</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="6732151329960766506" datatype="html">
|
<trans-unit id="6732151329960766506" datatype="html">
|
||||||
|
@ -303,12 +303,17 @@ describe('SettingsComponent', () => {
|
|||||||
redis_error:
|
redis_error:
|
||||||
'Error 61 connecting to localhost:6379. Connection refused.',
|
'Error 61 connecting to localhost:6379. Connection refused.',
|
||||||
celery_status: SystemStatusItemStatus.ERROR,
|
celery_status: SystemStatusItemStatus.ERROR,
|
||||||
|
celery_url: 'celery@localhost',
|
||||||
|
celery_error: 'Error connecting to celery@localhost',
|
||||||
index_status: SystemStatusItemStatus.OK,
|
index_status: SystemStatusItemStatus.OK,
|
||||||
index_last_modified: new Date().toISOString(),
|
index_last_modified: new Date().toISOString(),
|
||||||
index_error: null,
|
index_error: null,
|
||||||
classifier_status: SystemStatusItemStatus.OK,
|
classifier_status: SystemStatusItemStatus.OK,
|
||||||
classifier_last_trained: new Date().toISOString(),
|
classifier_last_trained: new Date().toISOString(),
|
||||||
classifier_error: null,
|
classifier_error: null,
|
||||||
|
sanity_check_status: SystemStatusItemStatus.ERROR,
|
||||||
|
sanity_check_last_run: new Date().toISOString(),
|
||||||
|
sanity_check_error: 'Error running sanity check.',
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
jest.spyOn(systemStatusService, 'get').mockReturnValue(of(status))
|
jest.spyOn(systemStatusService, 'get').mockReturnValue(of(status))
|
||||||
|
@ -19,6 +19,7 @@ import { allIcons, NgxBootstrapIconsModule } from 'ngx-bootstrap-icons'
|
|||||||
import { routes } from 'src/app/app-routing.module'
|
import { routes } from 'src/app/app-routing.module'
|
||||||
import {
|
import {
|
||||||
PaperlessTask,
|
PaperlessTask,
|
||||||
|
PaperlessTaskName,
|
||||||
PaperlessTaskStatus,
|
PaperlessTaskStatus,
|
||||||
PaperlessTaskType,
|
PaperlessTaskType,
|
||||||
} from 'src/app/data/paperless-task'
|
} from 'src/app/data/paperless-task'
|
||||||
@ -39,7 +40,8 @@ const tasks: PaperlessTask[] = [
|
|||||||
task_file_name: 'test.pdf',
|
task_file_name: 'test.pdf',
|
||||||
date_created: new Date('2023-03-01T10:26:03.093116Z'),
|
date_created: new Date('2023-03-01T10:26:03.093116Z'),
|
||||||
date_done: new Date('2023-03-01T10:26:07.223048Z'),
|
date_done: new Date('2023-03-01T10:26:07.223048Z'),
|
||||||
type: PaperlessTaskType.File,
|
type: PaperlessTaskType.Auto,
|
||||||
|
task_name: PaperlessTaskName.ConsumeFile,
|
||||||
status: PaperlessTaskStatus.Failed,
|
status: PaperlessTaskStatus.Failed,
|
||||||
result: 'test.pd: Not consuming test.pdf: It is a duplicate of test (#100)',
|
result: 'test.pd: Not consuming test.pdf: It is a duplicate of test (#100)',
|
||||||
acknowledged: false,
|
acknowledged: false,
|
||||||
@ -51,7 +53,8 @@ const tasks: PaperlessTask[] = [
|
|||||||
task_file_name: '191092.pdf',
|
task_file_name: '191092.pdf',
|
||||||
date_created: new Date('2023-03-01T09:26:03.093116Z'),
|
date_created: new Date('2023-03-01T09:26:03.093116Z'),
|
||||||
date_done: new Date('2023-03-01T09:26:07.223048Z'),
|
date_done: new Date('2023-03-01T09:26:07.223048Z'),
|
||||||
type: PaperlessTaskType.File,
|
type: PaperlessTaskType.Auto,
|
||||||
|
task_name: PaperlessTaskName.ConsumeFile,
|
||||||
status: PaperlessTaskStatus.Failed,
|
status: PaperlessTaskStatus.Failed,
|
||||||
result:
|
result:
|
||||||
'191092.pd: Not consuming 191092.pdf: It is a duplicate of 191092 (#311)',
|
'191092.pd: Not consuming 191092.pdf: It is a duplicate of 191092 (#311)',
|
||||||
@ -64,7 +67,8 @@ const tasks: PaperlessTask[] = [
|
|||||||
task_file_name: 'Scan Jun 6, 2023 at 3.19 PM.pdf',
|
task_file_name: 'Scan Jun 6, 2023 at 3.19 PM.pdf',
|
||||||
date_created: new Date('2023-06-06T15:22:05.722323-07:00'),
|
date_created: new Date('2023-06-06T15:22:05.722323-07:00'),
|
||||||
date_done: new Date('2023-06-06T15:22:14.564305-07:00'),
|
date_done: new Date('2023-06-06T15:22:14.564305-07:00'),
|
||||||
type: PaperlessTaskType.File,
|
type: PaperlessTaskType.Auto,
|
||||||
|
task_name: PaperlessTaskName.ConsumeFile,
|
||||||
status: PaperlessTaskStatus.Pending,
|
status: PaperlessTaskStatus.Pending,
|
||||||
result: null,
|
result: null,
|
||||||
acknowledged: false,
|
acknowledged: false,
|
||||||
@ -76,7 +80,8 @@ const tasks: PaperlessTask[] = [
|
|||||||
task_file_name: 'paperless-mail-l4dkg8ir',
|
task_file_name: 'paperless-mail-l4dkg8ir',
|
||||||
date_created: new Date('2023-06-04T11:24:32.898089-07:00'),
|
date_created: new Date('2023-06-04T11:24:32.898089-07:00'),
|
||||||
date_done: new Date('2023-06-04T11:24:44.678605-07:00'),
|
date_done: new Date('2023-06-04T11:24:44.678605-07:00'),
|
||||||
type: PaperlessTaskType.File,
|
type: PaperlessTaskType.Auto,
|
||||||
|
task_name: PaperlessTaskName.ConsumeFile,
|
||||||
status: PaperlessTaskStatus.Complete,
|
status: PaperlessTaskStatus.Complete,
|
||||||
result: 'Success. New document id 422 created',
|
result: 'Success. New document id 422 created',
|
||||||
acknowledged: false,
|
acknowledged: false,
|
||||||
@ -88,7 +93,8 @@ const tasks: PaperlessTask[] = [
|
|||||||
task_file_name: 'onlinePaymentSummary.pdf',
|
task_file_name: 'onlinePaymentSummary.pdf',
|
||||||
date_created: new Date('2023-06-01T13:49:51.631305-07:00'),
|
date_created: new Date('2023-06-01T13:49:51.631305-07:00'),
|
||||||
date_done: new Date('2023-06-01T13:49:54.190220-07:00'),
|
date_done: new Date('2023-06-01T13:49:54.190220-07:00'),
|
||||||
type: PaperlessTaskType.File,
|
type: PaperlessTaskType.Auto,
|
||||||
|
task_name: PaperlessTaskName.ConsumeFile,
|
||||||
status: PaperlessTaskStatus.Complete,
|
status: PaperlessTaskStatus.Complete,
|
||||||
result: 'Success. New document id 421 created',
|
result: 'Success. New document id 421 created',
|
||||||
acknowledged: false,
|
acknowledged: false,
|
||||||
@ -100,7 +106,8 @@ const tasks: PaperlessTask[] = [
|
|||||||
task_file_name: 'paperless-mail-_rrpmqk6',
|
task_file_name: 'paperless-mail-_rrpmqk6',
|
||||||
date_created: new Date('2023-06-07T02:54:35.694916Z'),
|
date_created: new Date('2023-06-07T02:54:35.694916Z'),
|
||||||
date_done: null,
|
date_done: null,
|
||||||
type: PaperlessTaskType.File,
|
type: PaperlessTaskType.Auto,
|
||||||
|
task_name: PaperlessTaskName.ConsumeFile,
|
||||||
status: PaperlessTaskStatus.Started,
|
status: PaperlessTaskStatus.Started,
|
||||||
result: null,
|
result: null,
|
||||||
acknowledged: false,
|
acknowledged: false,
|
||||||
@ -155,7 +162,9 @@ describe('TasksComponent', () => {
|
|||||||
jest.useFakeTimers()
|
jest.useFakeTimers()
|
||||||
fixture.detectChanges()
|
fixture.detectChanges()
|
||||||
httpTestingController
|
httpTestingController
|
||||||
.expectOne(`${environment.apiBaseUrl}tasks/`)
|
.expectOne(
|
||||||
|
`${environment.apiBaseUrl}tasks/?task_name=consume_file&acknowledged=false`
|
||||||
|
)
|
||||||
.flush(tasks)
|
.flush(tasks)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
<div class="modal-header">
|
<div class="modal-header">
|
||||||
<h5 class="modal-title" id="modal-basic-title" i18n>System Status</h5>
|
<h6 class="modal-title" id="modal-basic-title" i18n>System Status</h6>
|
||||||
<button type="button" class="btn-close" aria-label="Close" (click)="close()"></button>
|
<button type="button" class="btn-close" aria-label="Close" (click)="close()"></button>
|
||||||
</div>
|
</div>
|
||||||
<div class="modal-body">
|
<div class="modal-body">
|
||||||
@ -11,11 +11,11 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
} @else {
|
} @else {
|
||||||
<div class="row row-cols-1 row-cols-md-3 g-3">
|
<div class="row row-cols-1 row-cols-md-4 g-3">
|
||||||
<div class="col">
|
<div class="col-4">
|
||||||
<div class="card bg-light h-100">
|
<div class="card bg-light h-100">
|
||||||
<div class="card-header">
|
<div class="card-header">
|
||||||
<h5 class="card-title mb-0" i18n>Environment</h5>
|
<h6 class="card-title mb-0" i18n>Environment</h6>
|
||||||
</div>
|
</div>
|
||||||
<div class="card-body">
|
<div class="card-body">
|
||||||
<dl class="card-text">
|
<dl class="card-text">
|
||||||
@ -38,27 +38,37 @@
|
|||||||
<div class="col">
|
<div class="col">
|
||||||
<div class="card bg-light h-100">
|
<div class="card bg-light h-100">
|
||||||
<div class="card-header">
|
<div class="card-header">
|
||||||
<h5 class="card-title mb-0" i18n>Database</h5>
|
<h6 class="card-title mb-0" i18n>Database</h6>
|
||||||
</div>
|
</div>
|
||||||
<div class="card-body">
|
<div class="card-body">
|
||||||
<dl class="card-text">
|
<dl class="card-text">
|
||||||
<dt i18n>Type</dt>
|
<dt i18n>Type</dt>
|
||||||
<dd>{{status.database.type}}</dd>
|
<dd>{{status.database.type}}</dd>
|
||||||
<dt i18n>Status</dt>
|
<dt i18n>Status</dt>
|
||||||
<dd class="d-flex align-items-center">
|
<dd>
|
||||||
|
<div class="badge text-uppercase bg-info text-dark" [ngbPopover]="databaseStatus" triggers="mouseenter:mouseleave">
|
||||||
{{status.database.status}}
|
{{status.database.status}}
|
||||||
@if (status.database.status === 'OK') {
|
@if (status.database.status === 'OK') {
|
||||||
<i-bs name="check-circle-fill" class="text-primary ms-2 lh-1" ngbPopover="{{status.database.url}}" triggers="mouseenter:mouseleave"></i-bs>
|
<i-bs name="check-circle-fill" class="text-primary ms-2 lh-1"></i-bs>
|
||||||
} @else {
|
} @else {
|
||||||
<i-bs name="exclamation-triangle-fill" class="text-danger ms-2 lh-1" ngbPopover="{{status.database.url}}: {{status.database.error}}" triggers="mouseenter:mouseleave"></i-bs>
|
<i-bs name="exclamation-triangle-fill" class="text-danger ms-2 lh-1"></i-bs>
|
||||||
}
|
}
|
||||||
|
</div>
|
||||||
|
<ng-template #databaseStatus>
|
||||||
|
@if (status.database.status === 'OK') {
|
||||||
|
{{status.database.url}}
|
||||||
|
} @else {
|
||||||
|
{{status.database.url}}: {{status.database.error}}
|
||||||
|
}
|
||||||
|
</ng-template>
|
||||||
</dd>
|
</dd>
|
||||||
<dt i18n>Migration Status</dt>
|
<dt i18n>Migration Status</dt>
|
||||||
<dd class="d-flex align-items-center">
|
<dd>
|
||||||
|
<div class="badge text-uppercase bg-info text-dark" [ngbPopover]="migrationStatus" triggers="mouseenter:mouseleave">
|
||||||
@if (status.database.migration_status.unapplied_migrations.length === 0) {
|
@if (status.database.migration_status.unapplied_migrations.length === 0) {
|
||||||
<ng-container i18n>Up to date</ng-container><i-bs name="check-circle-fill" class="text-primary ms-2 lh-1" [ngbPopover]="migrationStatus" triggers="mouseenter:mouseleave"></i-bs>
|
<ng-container i18n>Up to date</ng-container><i-bs name="check-circle-fill" class="text-primary ms-2 lh-1"></i-bs>
|
||||||
} @else {
|
} @else {
|
||||||
<ng-container>{{status.database.migration_status.unapplied_migrations.length}} Pending</ng-container><i-bs name="exclamation-triangle-fill" class="text-warning ms-2 lh-1" [ngbPopover]="migrationStatus" triggers="mouseenter:mouseleave"></i-bs>
|
<ng-container>{{status.database.migration_status.unapplied_migrations.length}} Pending</ng-container><i-bs name="exclamation-triangle-fill" class="text-warning ms-2 lh-1"></i-bs>
|
||||||
}
|
}
|
||||||
<ng-template #migrationStatus>
|
<ng-template #migrationStatus>
|
||||||
<h6><ng-container i18n>Latest Migration</ng-container>:</h6> <span class="font-monospace small">{{status.database.migration_status.latest_migration}}</span>
|
<h6><ng-container i18n>Latest Migration</ng-container>:</h6> <span class="font-monospace small">{{status.database.migration_status.latest_migration}}</span>
|
||||||
@ -71,6 +81,7 @@
|
|||||||
</ul>
|
</ul>
|
||||||
}
|
}
|
||||||
</ng-template>
|
</ng-template>
|
||||||
|
</div>
|
||||||
</dd>
|
</dd>
|
||||||
</dl>
|
</dl>
|
||||||
</div>
|
</div>
|
||||||
@ -80,63 +91,127 @@
|
|||||||
<div class="col">
|
<div class="col">
|
||||||
<div class="card bg-light h-100">
|
<div class="card bg-light h-100">
|
||||||
<div class="card-header">
|
<div class="card-header">
|
||||||
<h5 class="card-title mb-0" i18n>Tasks</h5>
|
<h6 class="card-title mb-0" i18n>Tasks Queue</h6>
|
||||||
</div>
|
</div>
|
||||||
<div class="card-body">
|
<div class="card-body">
|
||||||
<dl class="card-text">
|
<dl class="card-text">
|
||||||
<dt i18n>Redis Status</dt>
|
<dt i18n>Redis Status</dt>
|
||||||
<dd class="d-flex align-items-center">
|
<dd>
|
||||||
|
<div class="badge text-uppercase bg-info text-dark" [ngbPopover]="redisStatus" triggers="mouseenter:mouseleave">
|
||||||
{{status.tasks.redis_status}}
|
{{status.tasks.redis_status}}
|
||||||
@if (status.tasks.redis_status === 'OK') {
|
@if (status.tasks.redis_status === 'OK') {
|
||||||
<i-bs name="check-circle-fill" class="text-primary ms-2 lh-1" ngbPopover="{{status.tasks.redis_url}}" triggers="mouseenter:mouseleave"></i-bs>
|
<i-bs name="check-circle-fill" class="text-primary ms-2 lh-1"></i-bs>
|
||||||
} @else {
|
} @else {
|
||||||
<i-bs name="exclamation-triangle-fill" class="text-danger ms-2 lh-1" ngbPopover="{{status.tasks.redis_url}}: {{status.tasks.redis_error}}" triggers="mouseenter:mouseleave"></i-bs>
|
<i-bs name="exclamation-triangle-fill" class="text-danger ms-2 lh-1"></i-bs>
|
||||||
}
|
}
|
||||||
|
</div>
|
||||||
|
<ng-template #redisStatus>
|
||||||
|
@if (status.tasks.redis_status === 'OK') {
|
||||||
|
{{status.tasks.redis_url}}
|
||||||
|
} @else {
|
||||||
|
{{status.tasks.redis_url}}: {{status.tasks.redis_error}}
|
||||||
|
}
|
||||||
|
</ng-template>
|
||||||
</dd>
|
</dd>
|
||||||
<dt i18n>Celery Status</dt>
|
<dt i18n>Celery Status</dt>
|
||||||
<dd class="d-flex align-items-center">
|
<dd>
|
||||||
|
<div class="badge text-uppercase bg-info text-dark" [ngbPopover]="celeryStatus" triggers="mouseenter:mouseleave">
|
||||||
{{status.tasks.celery_status}}
|
{{status.tasks.celery_status}}
|
||||||
@if (status.tasks.celery_status === 'OK') {
|
@if (status.tasks.celery_status === 'OK') {
|
||||||
<i-bs name="check-circle-fill" class="text-primary ms-2 lh-1"></i-bs>
|
<i-bs name="check-circle-fill" class="text-primary ms-2 lh-1"></i-bs>
|
||||||
} @else {
|
} @else {
|
||||||
<i-bs name="exclamation-triangle-fill" class="text-danger ms-2 lh-1"></i-bs>
|
<i-bs name="exclamation-triangle-fill" class="text-danger ms-2 lh-1"></i-bs>
|
||||||
}
|
}
|
||||||
|
</div>
|
||||||
|
<ng-template #celeryStatus>
|
||||||
|
@if (status.tasks.celery_status === 'OK') {
|
||||||
|
{{status.tasks.celery_url}}
|
||||||
|
} @else {
|
||||||
|
{{status.tasks.celery_error}}
|
||||||
|
}
|
||||||
|
</ng-template>
|
||||||
</dd>
|
</dd>
|
||||||
|
</dl>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="col">
|
||||||
|
<div class="card bg-light h-100">
|
||||||
|
<div class="card-header">
|
||||||
|
<h6 class="card-title mb-0" i18n>Health</h6>
|
||||||
|
</div>
|
||||||
|
<div class="card-body">
|
||||||
|
<dl class="card-text">
|
||||||
<dt i18n>Search Index</dt>
|
<dt i18n>Search Index</dt>
|
||||||
<dd class="d-flex align-items-center">
|
<dd>
|
||||||
|
<div class="badge text-uppercase bg-info text-dark" [ngbPopover]="indexStatus" triggers="mouseenter:mouseleave">
|
||||||
{{status.tasks.index_status}}
|
{{status.tasks.index_status}}
|
||||||
@if (status.tasks.index_status === 'OK') {
|
@if (status.tasks.index_status === 'OK') {
|
||||||
@if (isStale(status.tasks.index_last_modified)) {
|
@if (isStale(status.tasks.index_last_modified)) {
|
||||||
<i-bs name="exclamation-triangle-fill" class="text-warning ms-2 lh-1" [ngbPopover]="indexStatus" triggers="mouseenter:mouseleave"></i-bs>
|
<i-bs name="exclamation-triangle-fill" class="text-warning ms-2 lh-1"></i-bs>
|
||||||
} @else {
|
} @else {
|
||||||
<i-bs name="check-circle-fill" class="text-primary ms-2 lh-1" [ngbPopover]="indexStatus" triggers="mouseenter:mouseleave"></i-bs>
|
<i-bs name="check-circle-fill" class="text-primary ms-2 lh-1"></i-bs>
|
||||||
}
|
}
|
||||||
} @else {
|
} @else {
|
||||||
<i-bs name="exclamation-triangle-fill" class="text-danger ms-2 lh-1" ngbPopover="{{status.tasks.index_error}}" triggers="mouseenter:mouseleave"></i-bs>
|
<i-bs name="exclamation-triangle-fill" class="text-danger ms-2 lh-1"></i-bs>
|
||||||
}
|
}
|
||||||
|
</div>
|
||||||
</dd>
|
</dd>
|
||||||
<ng-template #indexStatus>
|
<ng-template #indexStatus>
|
||||||
|
@if (status.tasks.index_status === 'OK') {
|
||||||
<h6><ng-container i18n>Last Updated</ng-container>:</h6> <span class="font-monospace small">{{status.tasks.index_last_modified | customDate:'medium'}}</span>
|
<h6><ng-container i18n>Last Updated</ng-container>:</h6> <span class="font-monospace small">{{status.tasks.index_last_modified | customDate:'medium'}}</span>
|
||||||
|
} @else {
|
||||||
|
<h6><ng-container i18n>Error</ng-container>:</h6> <span class="font-monospace small">{{status.tasks.index_error}}</span>
|
||||||
|
}
|
||||||
</ng-template>
|
</ng-template>
|
||||||
<dt i18n>Classifier</dt>
|
<dt i18n>Classifier</dt>
|
||||||
<dd class="d-flex align-items-center">
|
<dd>
|
||||||
|
<div class="badge text-uppercase bg-info text-dark" [ngbPopover]="classifierStatus" triggers="mouseenter:mouseleave">
|
||||||
{{status.tasks.classifier_status}}
|
{{status.tasks.classifier_status}}
|
||||||
@if (status.tasks.classifier_status === 'OK') {
|
@if (status.tasks.classifier_status === 'OK') {
|
||||||
@if (isStale(status.tasks.classifier_last_trained)) {
|
@if (isStale(status.tasks.classifier_last_trained)) {
|
||||||
<i-bs name="exclamation-triangle-fill" class="text-warning ms-2 lh-1" [ngbPopover]="classifierStatus" triggers="mouseenter:mouseleave"></i-bs>
|
<i-bs name="exclamation-triangle-fill" class="text-warning ms-2 lh-1"></i-bs>
|
||||||
} @else {
|
} @else {
|
||||||
<i-bs name="check-circle-fill" class="text-primary ms-2 lh-1" [ngbPopover]="classifierStatus" triggers="mouseenter:mouseleave"></i-bs>
|
<i-bs name="check-circle-fill" class="text-primary ms-2 lh-1"></i-bs>
|
||||||
}
|
}
|
||||||
} @else {
|
} @else {
|
||||||
<i-bs name="exclamation-triangle-fill" class="ms-2 lh-1"
|
<i-bs name="exclamation-triangle-fill" class="ms-2 lh-1"
|
||||||
[class.text-danger]="status.tasks.classifier_status === SystemStatusItemStatus.ERROR"
|
[class.text-danger]="status.tasks.classifier_status === SystemStatusItemStatus.ERROR"
|
||||||
[class.text-warning]="status.tasks.classifier_status === SystemStatusItemStatus.WARNING"
|
[class.text-warning]="status.tasks.classifier_status === SystemStatusItemStatus.WARNING"></i-bs>
|
||||||
ngbPopover="{{status.tasks.classifier_error}}"
|
|
||||||
triggers="mouseenter:mouseleave"></i-bs>
|
|
||||||
}
|
}
|
||||||
|
</div>
|
||||||
</dd>
|
</dd>
|
||||||
<ng-template #classifierStatus>
|
<ng-template #classifierStatus>
|
||||||
|
@if (status.tasks.classifier_status === 'OK') {
|
||||||
<h6><ng-container i18n>Last Trained</ng-container>:</h6> <span class="font-monospace small">{{status.tasks.classifier_last_trained | customDate:'medium'}}</span>
|
<h6><ng-container i18n>Last Trained</ng-container>:</h6> <span class="font-monospace small">{{status.tasks.classifier_last_trained | customDate:'medium'}}</span>
|
||||||
|
} @else {
|
||||||
|
<h6><ng-container i18n>Error</ng-container>:</h6> <span class="font-monospace small">{{status.tasks.classifier_error}}</span>
|
||||||
|
}
|
||||||
|
</ng-template>
|
||||||
|
<dt i18n>Sanity Checker</dt>
|
||||||
|
<dd>
|
||||||
|
<div class="badge text-uppercase bg-info text-dark" [ngbPopover]="sanityCheckerStatus" triggers="mouseenter:mouseleave">
|
||||||
|
{{status.tasks.sanity_check_status}}
|
||||||
|
@if (status.tasks.sanity_check_status === 'OK') {
|
||||||
|
@if (isStale(status.tasks.sanity_check_last_run)) {
|
||||||
|
<i-bs name="exclamation-triangle-fill" class="text-warning ms-2 lh-1"></i-bs>
|
||||||
|
} @else {
|
||||||
|
<i-bs name="check-circle-fill" class="text-primary ms-2 lh-1"></i-bs>
|
||||||
|
}
|
||||||
|
} @else {
|
||||||
|
<i-bs name="exclamation-triangle-fill" class="ms-2 lh-1"
|
||||||
|
[class.text-danger]="status.tasks.sanity_check_status === SystemStatusItemStatus.ERROR"
|
||||||
|
[class.text-warning]="status.tasks.sanity_check_status === SystemStatusItemStatus.WARNING"></i-bs>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</dd>
|
||||||
|
<ng-template #sanityCheckerStatus>
|
||||||
|
@if (status.tasks.sanity_check_status === 'OK') {
|
||||||
|
<h6><ng-container i18n>Last Run</ng-container>:</h6> <span class="font-monospace small">{{status.tasks.sanity_check_last_run | customDate:'medium'}}</span>
|
||||||
|
} @else {
|
||||||
|
<h6><ng-container i18n>Error</ng-container>:</h6> <span class="font-monospace small">{{status.tasks.sanity_check_error}}</span>
|
||||||
|
}
|
||||||
</ng-template>
|
</ng-template>
|
||||||
</dl>
|
</dl>
|
||||||
</div>
|
</div>
|
||||||
|
@ -0,0 +1,3 @@
|
|||||||
|
.border-primary {
|
||||||
|
--bs-border-color: var(--bs-primary);
|
||||||
|
}
|
@ -36,12 +36,17 @@ const status: SystemStatus = {
|
|||||||
redis_status: SystemStatusItemStatus.ERROR,
|
redis_status: SystemStatusItemStatus.ERROR,
|
||||||
redis_error: 'Error 61 connecting to localhost:6379. Connection refused.',
|
redis_error: 'Error 61 connecting to localhost:6379. Connection refused.',
|
||||||
celery_status: SystemStatusItemStatus.ERROR,
|
celery_status: SystemStatusItemStatus.ERROR,
|
||||||
|
celery_url: 'celery@localhost',
|
||||||
|
celery_error: 'Error connecting to celery@localhost',
|
||||||
index_status: SystemStatusItemStatus.OK,
|
index_status: SystemStatusItemStatus.OK,
|
||||||
index_last_modified: new Date().toISOString(),
|
index_last_modified: new Date().toISOString(),
|
||||||
index_error: null,
|
index_error: null,
|
||||||
classifier_status: SystemStatusItemStatus.OK,
|
classifier_status: SystemStatusItemStatus.OK,
|
||||||
classifier_last_trained: new Date().toISOString(),
|
classifier_last_trained: new Date().toISOString(),
|
||||||
classifier_error: null,
|
classifier_error: null,
|
||||||
|
sanity_check_status: SystemStatusItemStatus.OK,
|
||||||
|
sanity_check_last_run: new Date().toISOString(),
|
||||||
|
sanity_check_error: null,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,8 +1,15 @@
|
|||||||
import { ObjectWithId } from './object-with-id'
|
import { ObjectWithId } from './object-with-id'
|
||||||
|
|
||||||
export enum PaperlessTaskType {
|
export enum PaperlessTaskType {
|
||||||
// just file tasks, for now
|
Auto = 'auto_task',
|
||||||
File = 'file',
|
ScheduledTask = 'scheduled_task',
|
||||||
|
ManualTask = 'manual_task',
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum PaperlessTaskName {
|
||||||
|
ConsumeFile = 'consume_file',
|
||||||
|
TrainClassifier = 'train_classifier',
|
||||||
|
SanityCheck = 'check_sanity',
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum PaperlessTaskStatus {
|
export enum PaperlessTaskStatus {
|
||||||
@ -23,6 +30,8 @@ export interface PaperlessTask extends ObjectWithId {
|
|||||||
|
|
||||||
task_file_name: string
|
task_file_name: string
|
||||||
|
|
||||||
|
task_name: PaperlessTaskName
|
||||||
|
|
||||||
date_created: Date
|
date_created: Date
|
||||||
|
|
||||||
date_done?: Date
|
date_done?: Date
|
||||||
|
@ -32,11 +32,16 @@ export interface SystemStatus {
|
|||||||
redis_status: SystemStatusItemStatus
|
redis_status: SystemStatusItemStatus
|
||||||
redis_error: string
|
redis_error: string
|
||||||
celery_status: SystemStatusItemStatus
|
celery_status: SystemStatusItemStatus
|
||||||
|
celery_url: string
|
||||||
|
celery_error: string
|
||||||
index_status: SystemStatusItemStatus
|
index_status: SystemStatusItemStatus
|
||||||
index_last_modified: string // ISO date string
|
index_last_modified: string // ISO date string
|
||||||
index_error: string
|
index_error: string
|
||||||
classifier_status: SystemStatusItemStatus
|
classifier_status: SystemStatusItemStatus
|
||||||
classifier_last_trained: string // ISO date string
|
classifier_last_trained: string // ISO date string
|
||||||
classifier_error: string
|
classifier_error: string
|
||||||
|
sanity_check_status: SystemStatusItemStatus
|
||||||
|
sanity_check_last_run: string // ISO date string
|
||||||
|
sanity_check_error: string
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,7 +5,11 @@ import {
|
|||||||
} from '@angular/common/http/testing'
|
} from '@angular/common/http/testing'
|
||||||
import { TestBed } from '@angular/core/testing'
|
import { TestBed } from '@angular/core/testing'
|
||||||
import { environment } from 'src/environments/environment'
|
import { environment } from 'src/environments/environment'
|
||||||
import { PaperlessTaskStatus, PaperlessTaskType } from '../data/paperless-task'
|
import {
|
||||||
|
PaperlessTaskName,
|
||||||
|
PaperlessTaskStatus,
|
||||||
|
PaperlessTaskType,
|
||||||
|
} from '../data/paperless-task'
|
||||||
import { TasksService } from './tasks.service'
|
import { TasksService } from './tasks.service'
|
||||||
|
|
||||||
describe('TasksService', () => {
|
describe('TasksService', () => {
|
||||||
@ -33,7 +37,7 @@ describe('TasksService', () => {
|
|||||||
it('calls tasks api endpoint on reload', () => {
|
it('calls tasks api endpoint on reload', () => {
|
||||||
tasksService.reload()
|
tasksService.reload()
|
||||||
const req = httpTestingController.expectOne(
|
const req = httpTestingController.expectOne(
|
||||||
`${environment.apiBaseUrl}tasks/`
|
`${environment.apiBaseUrl}tasks/?task_name=consume_file&acknowledged=false`
|
||||||
)
|
)
|
||||||
expect(req.request.method).toEqual('GET')
|
expect(req.request.method).toEqual('GET')
|
||||||
})
|
})
|
||||||
@ -41,7 +45,9 @@ describe('TasksService', () => {
|
|||||||
it('does not call tasks api endpoint on reload if already loading', () => {
|
it('does not call tasks api endpoint on reload if already loading', () => {
|
||||||
tasksService.loading = true
|
tasksService.loading = true
|
||||||
tasksService.reload()
|
tasksService.reload()
|
||||||
httpTestingController.expectNone(`${environment.apiBaseUrl}tasks/`)
|
httpTestingController.expectNone(
|
||||||
|
`${environment.apiBaseUrl}tasks/?task_name=consume_file&acknowledged=false`
|
||||||
|
)
|
||||||
})
|
})
|
||||||
|
|
||||||
it('calls acknowledge_tasks api endpoint on dismiss and reloads', () => {
|
it('calls acknowledge_tasks api endpoint on dismiss and reloads', () => {
|
||||||
@ -55,14 +61,19 @@ describe('TasksService', () => {
|
|||||||
})
|
})
|
||||||
req.flush([])
|
req.flush([])
|
||||||
// reload is then called
|
// reload is then called
|
||||||
httpTestingController.expectOne(`${environment.apiBaseUrl}tasks/`).flush([])
|
httpTestingController
|
||||||
|
.expectOne(
|
||||||
|
`${environment.apiBaseUrl}tasks/?task_name=consume_file&acknowledged=false`
|
||||||
|
)
|
||||||
|
.flush([])
|
||||||
})
|
})
|
||||||
|
|
||||||
it('sorts tasks returned from api', () => {
|
it('sorts tasks returned from api', () => {
|
||||||
expect(tasksService.total).toEqual(0)
|
expect(tasksService.total).toEqual(0)
|
||||||
const mockTasks = [
|
const mockTasks = [
|
||||||
{
|
{
|
||||||
type: PaperlessTaskType.File,
|
type: PaperlessTaskType.Auto,
|
||||||
|
task_name: PaperlessTaskName.ConsumeFile,
|
||||||
status: PaperlessTaskStatus.Complete,
|
status: PaperlessTaskStatus.Complete,
|
||||||
acknowledged: false,
|
acknowledged: false,
|
||||||
task_id: '1234',
|
task_id: '1234',
|
||||||
@ -70,7 +81,8 @@ describe('TasksService', () => {
|
|||||||
date_created: new Date(),
|
date_created: new Date(),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: PaperlessTaskType.File,
|
type: PaperlessTaskType.Auto,
|
||||||
|
task_name: PaperlessTaskName.ConsumeFile,
|
||||||
status: PaperlessTaskStatus.Failed,
|
status: PaperlessTaskStatus.Failed,
|
||||||
acknowledged: false,
|
acknowledged: false,
|
||||||
task_id: '1235',
|
task_id: '1235',
|
||||||
@ -78,7 +90,8 @@ describe('TasksService', () => {
|
|||||||
date_created: new Date(),
|
date_created: new Date(),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: PaperlessTaskType.File,
|
type: PaperlessTaskType.Auto,
|
||||||
|
task_name: PaperlessTaskName.ConsumeFile,
|
||||||
status: PaperlessTaskStatus.Pending,
|
status: PaperlessTaskStatus.Pending,
|
||||||
acknowledged: false,
|
acknowledged: false,
|
||||||
task_id: '1236',
|
task_id: '1236',
|
||||||
@ -86,7 +99,8 @@ describe('TasksService', () => {
|
|||||||
date_created: new Date(),
|
date_created: new Date(),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: PaperlessTaskType.File,
|
type: PaperlessTaskType.Auto,
|
||||||
|
task_name: PaperlessTaskName.ConsumeFile,
|
||||||
status: PaperlessTaskStatus.Started,
|
status: PaperlessTaskStatus.Started,
|
||||||
acknowledged: false,
|
acknowledged: false,
|
||||||
task_id: '1237',
|
task_id: '1237',
|
||||||
@ -94,7 +108,8 @@ describe('TasksService', () => {
|
|||||||
date_created: new Date(),
|
date_created: new Date(),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: PaperlessTaskType.File,
|
type: PaperlessTaskType.Auto,
|
||||||
|
task_name: PaperlessTaskName.ConsumeFile,
|
||||||
status: PaperlessTaskStatus.Complete,
|
status: PaperlessTaskStatus.Complete,
|
||||||
acknowledged: false,
|
acknowledged: false,
|
||||||
task_id: '1238',
|
task_id: '1238',
|
||||||
@ -106,7 +121,7 @@ describe('TasksService', () => {
|
|||||||
tasksService.reload()
|
tasksService.reload()
|
||||||
|
|
||||||
const req = httpTestingController.expectOne(
|
const req = httpTestingController.expectOne(
|
||||||
`${environment.apiBaseUrl}tasks/`
|
`${environment.apiBaseUrl}tasks/?task_name=consume_file&acknowledged=false`
|
||||||
)
|
)
|
||||||
|
|
||||||
req.flush(mockTasks)
|
req.flush(mockTasks)
|
||||||
|
@ -4,8 +4,8 @@ import { Subject } from 'rxjs'
|
|||||||
import { first, takeUntil } from 'rxjs/operators'
|
import { first, takeUntil } from 'rxjs/operators'
|
||||||
import {
|
import {
|
||||||
PaperlessTask,
|
PaperlessTask,
|
||||||
|
PaperlessTaskName,
|
||||||
PaperlessTaskStatus,
|
PaperlessTaskStatus,
|
||||||
PaperlessTaskType,
|
|
||||||
} from 'src/app/data/paperless-task'
|
} from 'src/app/data/paperless-task'
|
||||||
import { environment } from 'src/environments/environment'
|
import { environment } from 'src/environments/environment'
|
||||||
|
|
||||||
@ -54,10 +54,14 @@ export class TasksService {
|
|||||||
this.loading = true
|
this.loading = true
|
||||||
|
|
||||||
this.http
|
this.http
|
||||||
.get<PaperlessTask[]>(`${this.baseUrl}tasks/`)
|
.get<PaperlessTask[]>(
|
||||||
|
`${this.baseUrl}tasks/?task_name=consume_file&acknowledged=false`
|
||||||
|
)
|
||||||
.pipe(takeUntil(this.unsubscribeNotifer), first())
|
.pipe(takeUntil(this.unsubscribeNotifer), first())
|
||||||
.subscribe((r) => {
|
.subscribe((r) => {
|
||||||
this.fileTasks = r.filter((t) => t.type == PaperlessTaskType.File) // they're all File tasks, for now
|
this.fileTasks = r.filter(
|
||||||
|
(t) => t.task_name == PaperlessTaskName.ConsumeFile
|
||||||
|
)
|
||||||
this.loading = false
|
this.loading = false
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -21,10 +21,12 @@
|
|||||||
--pngx-success-darken-10: hsl(152, 69%, 11%); // based on success #198754
|
--pngx-success-darken-10: hsl(152, 69%, 11%); // based on success #198754
|
||||||
--pngx-bg-alt: #fff;
|
--pngx-bg-alt: #fff;
|
||||||
--pngx-bg-darker: var(--bs-gray-100);
|
--pngx-bg-darker: var(--bs-gray-100);
|
||||||
--pngx-bg-alt2: var(--bs-gray-200);
|
--pngx-bg-alt2: var(--bs-gray-200); // #e9ecef
|
||||||
--pngx-bg-disabled: #f7f7f7;
|
--pngx-bg-disabled: #f7f7f7;
|
||||||
--pngx-focus-alpha: 0.3;
|
--pngx-focus-alpha: 0.3;
|
||||||
--pngx-toast-max-width: 360px;
|
--pngx-toast-max-width: 360px;
|
||||||
|
--bs-info: var(--pngx-bg-alt2);
|
||||||
|
--bs-info-rgb: 233, 236, 239;
|
||||||
@media screen and (min-width: 1024px) {
|
@media screen and (min-width: 1024px) {
|
||||||
--pngx-toast-max-width: 450px;
|
--pngx-toast-max-width: 450px;
|
||||||
}
|
}
|
||||||
@ -71,8 +73,15 @@ $form-check-radio-checked-bg-image-dark: url("data:image/svg+xml,<svg xmlns='htt
|
|||||||
}
|
}
|
||||||
|
|
||||||
@mixin dark-mode {
|
@mixin dark-mode {
|
||||||
--bs-body-color: #{$text-color-dark-bg};
|
|
||||||
--pngx-body-color-accent: #{$text-color-dark-bg-accent};
|
--pngx-body-color-accent: #{$text-color-dark-bg-accent};
|
||||||
|
--pngx-bg-alt: #242529;
|
||||||
|
--pngx-bg-alt2: #232323;
|
||||||
|
--pngx-bg-darker: #101216;
|
||||||
|
--pngx-bg-disabled: var(--pngx-bg-alt);
|
||||||
|
--pngx-focus-alpha: 0.6;
|
||||||
|
--pngx-primary-faded: var(--pngx-primary-darken-15);
|
||||||
|
--pngx-primary-text-contrast: var(--bs-body-color);
|
||||||
|
--bs-body-color: #{$text-color-dark-bg};
|
||||||
--bs-secondary-color: #6c757d;
|
--bs-secondary-color: #6c757d;
|
||||||
--bs-danger: #b71631;
|
--bs-danger: #b71631;
|
||||||
--bs-danger-rgb: 183, 22, 49;
|
--bs-danger-rgb: 183, 22, 49;
|
||||||
@ -80,15 +89,10 @@ $form-check-radio-checked-bg-image-dark: url("data:image/svg+xml,<svg xmlns='htt
|
|||||||
--bs-body-bg-rgb: 22, 22, 24;
|
--bs-body-bg-rgb: 22, 22, 24;
|
||||||
--bs-light: #1c1c1f;
|
--bs-light: #1c1c1f;
|
||||||
--bs-light-rgb: 28, 28, 31;
|
--bs-light-rgb: 28, 28, 31;
|
||||||
|
--bs-info: var(--pngx-bg-alt);
|
||||||
|
--bs-info-rgb: 36, 36, 39;
|
||||||
--bs-border-color: #47494f;
|
--bs-border-color: #47494f;
|
||||||
--pngx-bg-alt2: #232323;
|
|
||||||
--pngx-bg-darker: #101216;
|
|
||||||
--bs-tertiary-bg: var(--pngx-bg-darker);
|
--bs-tertiary-bg: var(--pngx-bg-darker);
|
||||||
--pngx-bg-alt: #242529;
|
|
||||||
--pngx-bg-disabled: var(--pngx-bg-alt);
|
|
||||||
--pngx-focus-alpha: 0.6;
|
|
||||||
--pngx-primary-faded: var(--pngx-primary-darken-15);
|
|
||||||
--pngx-primary-text-contrast: var(--bs-body-color);
|
|
||||||
--bs-dark-border-subtle: var(--pngx-bg-darker);
|
--bs-dark-border-subtle: var(--pngx-bg-darker);
|
||||||
--bs-border-color-translucent: rgba(0, 0, 0, .175); // override bs
|
--bs-border-color-translucent: rgba(0, 0, 0, .175); // override bs
|
||||||
|
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
import logging
|
import logging
|
||||||
import pickle
|
import pickle
|
||||||
import re
|
import re
|
||||||
import time
|
|
||||||
import warnings
|
import warnings
|
||||||
from collections.abc import Iterator
|
from collections.abc import Iterator
|
||||||
from hashlib import sha256
|
from hashlib import sha256
|
||||||
@ -142,19 +141,6 @@ class DocumentClassifier:
|
|||||||
):
|
):
|
||||||
raise IncompatibleClassifierVersionError("sklearn version update")
|
raise IncompatibleClassifierVersionError("sklearn version update")
|
||||||
|
|
||||||
def set_last_checked(self) -> None:
|
|
||||||
# save a timestamp of the last time we checked for retraining to a file
|
|
||||||
with Path(settings.MODEL_FILE.with_suffix(".last_checked")).open("w") as f:
|
|
||||||
f.write(str(time.time()))
|
|
||||||
|
|
||||||
def get_last_checked(self) -> float | None:
|
|
||||||
# load the timestamp of the last time we checked for retraining
|
|
||||||
try:
|
|
||||||
with Path(settings.MODEL_FILE.with_suffix(".last_checked")).open("r") as f:
|
|
||||||
return float(f.read())
|
|
||||||
except FileNotFoundError: # pragma: no cover
|
|
||||||
return None
|
|
||||||
|
|
||||||
def save(self) -> None:
|
def save(self) -> None:
|
||||||
target_file: Path = settings.MODEL_FILE
|
target_file: Path = settings.MODEL_FILE
|
||||||
target_file_temp: Path = target_file.with_suffix(".pickle.part")
|
target_file_temp: Path = target_file.with_suffix(".pickle.part")
|
||||||
@ -175,7 +161,6 @@ class DocumentClassifier:
|
|||||||
pickle.dump(self.storage_path_classifier, f)
|
pickle.dump(self.storage_path_classifier, f)
|
||||||
|
|
||||||
target_file_temp.rename(target_file)
|
target_file_temp.rename(target_file)
|
||||||
self.set_last_checked()
|
|
||||||
|
|
||||||
def train(self) -> bool:
|
def train(self) -> bool:
|
||||||
# Get non-inbox documents
|
# Get non-inbox documents
|
||||||
@ -244,7 +229,6 @@ class DocumentClassifier:
|
|||||||
and self.last_doc_change_time >= latest_doc_change
|
and self.last_doc_change_time >= latest_doc_change
|
||||||
) and self.last_auto_type_hash == hasher.digest():
|
) and self.last_auto_type_hash == hasher.digest():
|
||||||
logger.info("No updates since last training")
|
logger.info("No updates since last training")
|
||||||
self.set_last_checked()
|
|
||||||
# Set the classifier information into the cache
|
# Set the classifier information into the cache
|
||||||
# Caching for 50 minutes, so slightly less than the normal retrain time
|
# Caching for 50 minutes, so slightly less than the normal retrain time
|
||||||
cache.set(
|
cache.set(
|
||||||
|
@ -35,6 +35,7 @@ from documents.models import CustomFieldInstance
|
|||||||
from documents.models import Document
|
from documents.models import Document
|
||||||
from documents.models import DocumentType
|
from documents.models import DocumentType
|
||||||
from documents.models import Log
|
from documents.models import Log
|
||||||
|
from documents.models import PaperlessTask
|
||||||
from documents.models import ShareLink
|
from documents.models import ShareLink
|
||||||
from documents.models import StoragePath
|
from documents.models import StoragePath
|
||||||
from documents.models import Tag
|
from documents.models import Tag
|
||||||
@ -770,6 +771,21 @@ class ShareLinkFilterSet(FilterSet):
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class PaperlessTaskFilterSet(FilterSet):
|
||||||
|
acknowledged = BooleanFilter(
|
||||||
|
label="Acknowledged",
|
||||||
|
field_name="acknowledged",
|
||||||
|
)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = PaperlessTask
|
||||||
|
fields = {
|
||||||
|
"type": ["exact"],
|
||||||
|
"task_name": ["exact"],
|
||||||
|
"status": ["exact"],
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
class ObjectOwnedOrGrantedPermissionsFilter(ObjectPermissionsFilter):
|
class ObjectOwnedOrGrantedPermissionsFilter(ObjectPermissionsFilter):
|
||||||
"""
|
"""
|
||||||
A filter backend that limits results to those where the requesting user
|
A filter backend that limits results to those where the requesting user
|
||||||
|
@ -10,4 +10,4 @@ class Command(BaseCommand):
|
|||||||
)
|
)
|
||||||
|
|
||||||
def handle(self, *args, **options):
|
def handle(self, *args, **options):
|
||||||
train_classifier()
|
train_classifier(scheduled=False)
|
||||||
|
@ -12,6 +12,6 @@ class Command(ProgressBarMixin, BaseCommand):
|
|||||||
|
|
||||||
def handle(self, *args, **options):
|
def handle(self, *args, **options):
|
||||||
self.handle_progress_bar_mixin(**options)
|
self.handle_progress_bar_mixin(**options)
|
||||||
messages = check_sanity(progress=self.use_progress_bar)
|
messages = check_sanity(progress=self.use_progress_bar, scheduled=False)
|
||||||
|
|
||||||
messages.log_messages()
|
messages.log_messages()
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
# Generated by Django 5.1.6 on 2025-02-20 04:55
|
# Generated by Django 5.1.6 on 2025-02-21 16:34
|
||||||
|
|
||||||
import multiselectfield.db.fields
|
import multiselectfield.db.fields
|
||||||
from django.db import migrations
|
from django.db import migrations
|
||||||
@ -16,12 +16,51 @@ def update_workflow_sources(apps, schema_editor):
|
|||||||
trigger.save()
|
trigger.save()
|
||||||
|
|
||||||
|
|
||||||
|
def make_existing_tasks_consume_auto(apps, schema_editor):
|
||||||
|
PaperlessTask = apps.get_model("documents", "PaperlessTask")
|
||||||
|
PaperlessTask.objects.all().update(type="auto_task", task_name="consume_file")
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
class Migration(migrations.Migration):
|
||||||
dependencies = [
|
dependencies = [
|
||||||
("documents", "1062_alter_savedviewfilterrule_rule_type"),
|
("documents", "1062_alter_savedviewfilterrule_rule_type"),
|
||||||
]
|
]
|
||||||
|
|
||||||
operations = [
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="paperlesstask",
|
||||||
|
name="type",
|
||||||
|
field=models.CharField(
|
||||||
|
choices=[
|
||||||
|
("auto_task", "Auto Task"),
|
||||||
|
("scheduled_task", "Scheduled Task"),
|
||||||
|
("manual_task", "Manual Task"),
|
||||||
|
],
|
||||||
|
default="auto_task",
|
||||||
|
help_text="The type of task that was run",
|
||||||
|
max_length=30,
|
||||||
|
verbose_name="Task Type",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="paperlesstask",
|
||||||
|
name="task_name",
|
||||||
|
field=models.CharField(
|
||||||
|
choices=[
|
||||||
|
("consume_file", "Consume File"),
|
||||||
|
("train_classifier", "Train Classifier"),
|
||||||
|
("check_sanity", "Check Sanity"),
|
||||||
|
],
|
||||||
|
help_text="Name of the task that was run",
|
||||||
|
max_length=255,
|
||||||
|
null=True,
|
||||||
|
verbose_name="Task Name",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.RunPython(
|
||||||
|
code=make_existing_tasks_consume_auto,
|
||||||
|
reverse_code=migrations.RunPython.noop,
|
||||||
|
),
|
||||||
migrations.AlterField(
|
migrations.AlterField(
|
||||||
model_name="workflowactionwebhook",
|
model_name="workflowactionwebhook",
|
||||||
name="url",
|
name="url",
|
@ -650,6 +650,16 @@ class PaperlessTask(ModelWithOwner):
|
|||||||
ALL_STATES = sorted(states.ALL_STATES)
|
ALL_STATES = sorted(states.ALL_STATES)
|
||||||
TASK_STATE_CHOICES = sorted(zip(ALL_STATES, ALL_STATES))
|
TASK_STATE_CHOICES = sorted(zip(ALL_STATES, ALL_STATES))
|
||||||
|
|
||||||
|
class TaskType(models.TextChoices):
|
||||||
|
AUTO = ("auto_task", _("Auto Task"))
|
||||||
|
SCHEDULED_TASK = ("scheduled_task", _("Scheduled Task"))
|
||||||
|
MANUAL_TASK = ("manual_task", _("Manual Task"))
|
||||||
|
|
||||||
|
class TaskName(models.TextChoices):
|
||||||
|
CONSUME_FILE = ("consume_file", _("Consume File"))
|
||||||
|
TRAIN_CLASSIFIER = ("train_classifier", _("Train Classifier"))
|
||||||
|
CHECK_SANITY = ("check_sanity", _("Check Sanity"))
|
||||||
|
|
||||||
task_id = models.CharField(
|
task_id = models.CharField(
|
||||||
max_length=255,
|
max_length=255,
|
||||||
unique=True,
|
unique=True,
|
||||||
@ -673,8 +683,9 @@ class PaperlessTask(ModelWithOwner):
|
|||||||
task_name = models.CharField(
|
task_name = models.CharField(
|
||||||
null=True,
|
null=True,
|
||||||
max_length=255,
|
max_length=255,
|
||||||
|
choices=TaskName.choices,
|
||||||
verbose_name=_("Task Name"),
|
verbose_name=_("Task Name"),
|
||||||
help_text=_("Name of the Task which was run"),
|
help_text=_("Name of the task that was run"),
|
||||||
)
|
)
|
||||||
|
|
||||||
status = models.CharField(
|
status = models.CharField(
|
||||||
@ -684,24 +695,28 @@ class PaperlessTask(ModelWithOwner):
|
|||||||
verbose_name=_("Task State"),
|
verbose_name=_("Task State"),
|
||||||
help_text=_("Current state of the task being run"),
|
help_text=_("Current state of the task being run"),
|
||||||
)
|
)
|
||||||
|
|
||||||
date_created = models.DateTimeField(
|
date_created = models.DateTimeField(
|
||||||
null=True,
|
null=True,
|
||||||
default=timezone.now,
|
default=timezone.now,
|
||||||
verbose_name=_("Created DateTime"),
|
verbose_name=_("Created DateTime"),
|
||||||
help_text=_("Datetime field when the task result was created in UTC"),
|
help_text=_("Datetime field when the task result was created in UTC"),
|
||||||
)
|
)
|
||||||
|
|
||||||
date_started = models.DateTimeField(
|
date_started = models.DateTimeField(
|
||||||
null=True,
|
null=True,
|
||||||
default=None,
|
default=None,
|
||||||
verbose_name=_("Started DateTime"),
|
verbose_name=_("Started DateTime"),
|
||||||
help_text=_("Datetime field when the task was started in UTC"),
|
help_text=_("Datetime field when the task was started in UTC"),
|
||||||
)
|
)
|
||||||
|
|
||||||
date_done = models.DateTimeField(
|
date_done = models.DateTimeField(
|
||||||
null=True,
|
null=True,
|
||||||
default=None,
|
default=None,
|
||||||
verbose_name=_("Completed DateTime"),
|
verbose_name=_("Completed DateTime"),
|
||||||
help_text=_("Datetime field when the task was completed in UTC"),
|
help_text=_("Datetime field when the task was completed in UTC"),
|
||||||
)
|
)
|
||||||
|
|
||||||
result = models.TextField(
|
result = models.TextField(
|
||||||
null=True,
|
null=True,
|
||||||
default=None,
|
default=None,
|
||||||
@ -711,6 +726,14 @@ class PaperlessTask(ModelWithOwner):
|
|||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type = models.CharField(
|
||||||
|
max_length=30,
|
||||||
|
choices=TaskType.choices,
|
||||||
|
default=TaskType.AUTO,
|
||||||
|
verbose_name=_("Task Type"),
|
||||||
|
help_text=_("The type of task that was run"),
|
||||||
|
)
|
||||||
|
|
||||||
def __str__(self) -> str:
|
def __str__(self) -> str:
|
||||||
return f"Task {self.task_id}"
|
return f"Task {self.task_id}"
|
||||||
|
|
||||||
|
@ -1,13 +1,17 @@
|
|||||||
import hashlib
|
import hashlib
|
||||||
import logging
|
import logging
|
||||||
|
import uuid
|
||||||
from collections import defaultdict
|
from collections import defaultdict
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from typing import Final
|
from typing import Final
|
||||||
|
|
||||||
|
from celery import states
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
|
from django.utils import timezone
|
||||||
from tqdm import tqdm
|
from tqdm import tqdm
|
||||||
|
|
||||||
from documents.models import Document
|
from documents.models import Document
|
||||||
|
from documents.models import PaperlessTask
|
||||||
|
|
||||||
|
|
||||||
class SanityCheckMessages:
|
class SanityCheckMessages:
|
||||||
@ -57,7 +61,17 @@ class SanityCheckFailedException(Exception):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
def check_sanity(*, progress=False) -> SanityCheckMessages:
|
def check_sanity(*, progress=False, scheduled=True) -> SanityCheckMessages:
|
||||||
|
paperless_task = PaperlessTask.objects.create(
|
||||||
|
task_id=uuid.uuid4(),
|
||||||
|
type=PaperlessTask.TaskType.SCHEDULED_TASK
|
||||||
|
if scheduled
|
||||||
|
else PaperlessTask.TaskType.MANUAL_TASK,
|
||||||
|
task_name=PaperlessTask.TaskName.CHECK_SANITY,
|
||||||
|
status=states.STARTED,
|
||||||
|
date_created=timezone.now(),
|
||||||
|
date_started=timezone.now(),
|
||||||
|
)
|
||||||
messages = SanityCheckMessages()
|
messages = SanityCheckMessages()
|
||||||
|
|
||||||
present_files = {
|
present_files = {
|
||||||
@ -142,4 +156,11 @@ def check_sanity(*, progress=False) -> SanityCheckMessages:
|
|||||||
for extra_file in present_files:
|
for extra_file in present_files:
|
||||||
messages.warning(None, f"Orphaned file in media dir: {extra_file}")
|
messages.warning(None, f"Orphaned file in media dir: {extra_file}")
|
||||||
|
|
||||||
|
paperless_task.status = states.SUCCESS if not messages.has_error else states.FAILURE
|
||||||
|
# result is concatenated messages
|
||||||
|
paperless_task.result = f"{len(messages)} issues found."
|
||||||
|
if messages.has_error:
|
||||||
|
paperless_task.result += " Check logs for details."
|
||||||
|
paperless_task.date_done = timezone.now()
|
||||||
|
paperless_task.save(update_fields=["status", "result", "date_done"])
|
||||||
return messages
|
return messages
|
||||||
|
@ -1704,6 +1704,7 @@ class TasksViewSerializer(OwnedObjectSerializer):
|
|||||||
fields = (
|
fields = (
|
||||||
"id",
|
"id",
|
||||||
"task_id",
|
"task_id",
|
||||||
|
"task_name",
|
||||||
"task_file_name",
|
"task_file_name",
|
||||||
"date_created",
|
"date_created",
|
||||||
"date_done",
|
"date_done",
|
||||||
@ -1715,12 +1716,6 @@ class TasksViewSerializer(OwnedObjectSerializer):
|
|||||||
"owner",
|
"owner",
|
||||||
)
|
)
|
||||||
|
|
||||||
type = serializers.SerializerMethodField()
|
|
||||||
|
|
||||||
def get_type(self, obj) -> str:
|
|
||||||
# just file tasks, for now
|
|
||||||
return "file"
|
|
||||||
|
|
||||||
related_document = serializers.SerializerMethodField()
|
related_document = serializers.SerializerMethodField()
|
||||||
created_doc_re = re.compile(r"New document id (\d+) created")
|
created_doc_re = re.compile(r"New document id (\d+) created")
|
||||||
duplicate_doc_re = re.compile(r"It is a duplicate of .* \(#(\d+)\)")
|
duplicate_doc_re = re.compile(r"It is a duplicate of .* \(#(\d+)\)")
|
||||||
@ -1728,6 +1723,7 @@ class TasksViewSerializer(OwnedObjectSerializer):
|
|||||||
def get_related_document(self, obj) -> str | None:
|
def get_related_document(self, obj) -> str | None:
|
||||||
result = None
|
result = None
|
||||||
re = None
|
re = None
|
||||||
|
if obj.result:
|
||||||
match obj.status:
|
match obj.status:
|
||||||
case states.SUCCESS:
|
case states.SUCCESS:
|
||||||
re = self.created_doc_re
|
re = self.created_doc_re
|
||||||
|
@ -1221,10 +1221,11 @@ def before_task_publish_handler(sender=None, headers=None, body=None, **kwargs):
|
|||||||
user_id = overrides.owner_id if overrides else None
|
user_id = overrides.owner_id if overrides else None
|
||||||
|
|
||||||
PaperlessTask.objects.create(
|
PaperlessTask.objects.create(
|
||||||
|
type=PaperlessTask.TaskType.AUTO,
|
||||||
task_id=headers["id"],
|
task_id=headers["id"],
|
||||||
status=states.PENDING,
|
status=states.PENDING,
|
||||||
task_file_name=task_file_name,
|
task_file_name=task_file_name,
|
||||||
task_name=headers["task"],
|
task_name=PaperlessTask.TaskName.CONSUME_FILE,
|
||||||
result=None,
|
result=None,
|
||||||
date_created=timezone.now(),
|
date_created=timezone.now(),
|
||||||
date_started=None,
|
date_started=None,
|
||||||
|
@ -9,6 +9,7 @@ from tempfile import TemporaryDirectory
|
|||||||
import tqdm
|
import tqdm
|
||||||
from celery import Task
|
from celery import Task
|
||||||
from celery import shared_task
|
from celery import shared_task
|
||||||
|
from celery import states
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.contrib.contenttypes.models import ContentType
|
from django.contrib.contenttypes.models import ContentType
|
||||||
from django.db import models
|
from django.db import models
|
||||||
@ -35,6 +36,7 @@ from documents.models import Correspondent
|
|||||||
from documents.models import CustomFieldInstance
|
from documents.models import CustomFieldInstance
|
||||||
from documents.models import Document
|
from documents.models import Document
|
||||||
from documents.models import DocumentType
|
from documents.models import DocumentType
|
||||||
|
from documents.models import PaperlessTask
|
||||||
from documents.models import StoragePath
|
from documents.models import StoragePath
|
||||||
from documents.models import Tag
|
from documents.models import Tag
|
||||||
from documents.models import Workflow
|
from documents.models import Workflow
|
||||||
@ -74,19 +76,34 @@ def index_reindex(*, progress_bar_disable=False):
|
|||||||
|
|
||||||
|
|
||||||
@shared_task
|
@shared_task
|
||||||
def train_classifier():
|
def train_classifier(*, scheduled=True):
|
||||||
|
task = PaperlessTask.objects.create(
|
||||||
|
type=PaperlessTask.TaskType.SCHEDULED_TASK
|
||||||
|
if scheduled
|
||||||
|
else PaperlessTask.TaskType.MANUAL_TASK,
|
||||||
|
task_id=uuid.uuid4(),
|
||||||
|
task_name=PaperlessTask.TaskName.TRAIN_CLASSIFIER,
|
||||||
|
status=states.STARTED,
|
||||||
|
date_created=timezone.now(),
|
||||||
|
date_started=timezone.now(),
|
||||||
|
)
|
||||||
if (
|
if (
|
||||||
not Tag.objects.filter(matching_algorithm=Tag.MATCH_AUTO).exists()
|
not Tag.objects.filter(matching_algorithm=Tag.MATCH_AUTO).exists()
|
||||||
and not DocumentType.objects.filter(matching_algorithm=Tag.MATCH_AUTO).exists()
|
and not DocumentType.objects.filter(matching_algorithm=Tag.MATCH_AUTO).exists()
|
||||||
and not Correspondent.objects.filter(matching_algorithm=Tag.MATCH_AUTO).exists()
|
and not Correspondent.objects.filter(matching_algorithm=Tag.MATCH_AUTO).exists()
|
||||||
and not StoragePath.objects.filter(matching_algorithm=Tag.MATCH_AUTO).exists()
|
and not StoragePath.objects.filter(matching_algorithm=Tag.MATCH_AUTO).exists()
|
||||||
):
|
):
|
||||||
logger.info("No automatic matching items, not training")
|
result = "No automatic matching items, not training"
|
||||||
|
logger.info(result)
|
||||||
# Special case, items were once auto and trained, so remove the model
|
# Special case, items were once auto and trained, so remove the model
|
||||||
# and prevent its use again
|
# and prevent its use again
|
||||||
if settings.MODEL_FILE.exists():
|
if settings.MODEL_FILE.exists():
|
||||||
logger.info(f"Removing {settings.MODEL_FILE} so it won't be used")
|
logger.info(f"Removing {settings.MODEL_FILE} so it won't be used")
|
||||||
settings.MODEL_FILE.unlink()
|
settings.MODEL_FILE.unlink()
|
||||||
|
task.status = states.SUCCESS
|
||||||
|
task.result = result
|
||||||
|
task.date_done = timezone.now()
|
||||||
|
task.save()
|
||||||
return
|
return
|
||||||
|
|
||||||
classifier = load_classifier()
|
classifier = load_classifier()
|
||||||
@ -100,11 +117,19 @@ def train_classifier():
|
|||||||
f"Saving updated classifier model to {settings.MODEL_FILE}...",
|
f"Saving updated classifier model to {settings.MODEL_FILE}...",
|
||||||
)
|
)
|
||||||
classifier.save()
|
classifier.save()
|
||||||
|
task.result = "Training completed successfully"
|
||||||
else:
|
else:
|
||||||
logger.debug("Training data unchanged.")
|
logger.debug("Training data unchanged.")
|
||||||
|
task.result = "Training data unchanged"
|
||||||
|
|
||||||
|
task.status = states.SUCCESS
|
||||||
|
task.date_done = timezone.now()
|
||||||
|
task.save(update_fields=["status", "result", "date_done"])
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logger.warning("Classifier error: " + str(e))
|
logger.warning("Classifier error: " + str(e))
|
||||||
|
task.status = states.FAILURE
|
||||||
|
task.result = str(e)
|
||||||
|
|
||||||
|
|
||||||
@shared_task(bind=True)
|
@shared_task(bind=True)
|
||||||
|
@ -1,18 +1,14 @@
|
|||||||
import os
|
import os
|
||||||
import tempfile
|
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from unittest import mock
|
from unittest import mock
|
||||||
|
|
||||||
|
from celery import states
|
||||||
from django.contrib.auth.models import User
|
from django.contrib.auth.models import User
|
||||||
from django.test import override_settings
|
from django.test import override_settings
|
||||||
from rest_framework import status
|
from rest_framework import status
|
||||||
from rest_framework.test import APITestCase
|
from rest_framework.test import APITestCase
|
||||||
|
|
||||||
from documents.classifier import ClassifierModelCorruptError
|
from documents.models import PaperlessTask
|
||||||
from documents.classifier import DocumentClassifier
|
|
||||||
from documents.classifier import load_classifier
|
|
||||||
from documents.models import Document
|
|
||||||
from documents.models import Tag
|
|
||||||
from paperless import version
|
from paperless import version
|
||||||
|
|
||||||
|
|
||||||
@ -193,7 +189,6 @@ class TestSystemStatus(APITestCase):
|
|||||||
self.assertEqual(response.data["tasks"]["index_status"], "ERROR")
|
self.assertEqual(response.data["tasks"]["index_status"], "ERROR")
|
||||||
self.assertIsNotNone(response.data["tasks"]["index_error"])
|
self.assertIsNotNone(response.data["tasks"]["index_error"])
|
||||||
|
|
||||||
@override_settings(DATA_DIR=Path("/tmp/does_not_exist/data/"))
|
|
||||||
def test_system_status_classifier_ok(self):
|
def test_system_status_classifier_ok(self):
|
||||||
"""
|
"""
|
||||||
GIVEN:
|
GIVEN:
|
||||||
@ -203,9 +198,11 @@ class TestSystemStatus(APITestCase):
|
|||||||
THEN:
|
THEN:
|
||||||
- The response contains an OK classifier status
|
- The response contains an OK classifier status
|
||||||
"""
|
"""
|
||||||
load_classifier()
|
PaperlessTask.objects.create(
|
||||||
test_classifier = DocumentClassifier()
|
type=PaperlessTask.TaskType.SCHEDULED_TASK,
|
||||||
test_classifier.save()
|
status=states.SUCCESS,
|
||||||
|
task_name=PaperlessTask.TaskName.TRAIN_CLASSIFIER,
|
||||||
|
)
|
||||||
self.client.force_login(self.user)
|
self.client.force_login(self.user)
|
||||||
response = self.client.get(self.ENDPOINT)
|
response = self.client.get(self.ENDPOINT)
|
||||||
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
||||||
@ -215,51 +212,34 @@ class TestSystemStatus(APITestCase):
|
|||||||
def test_system_status_classifier_warning(self):
|
def test_system_status_classifier_warning(self):
|
||||||
"""
|
"""
|
||||||
GIVEN:
|
GIVEN:
|
||||||
- The classifier does not exist yet
|
- No classifier task is found
|
||||||
- > 0 documents and tags with auto matching exist
|
|
||||||
WHEN:
|
WHEN:
|
||||||
- The user requests the system status
|
- The user requests the system status
|
||||||
THEN:
|
THEN:
|
||||||
- The response contains an WARNING classifier status
|
- The response contains a WARNING classifier status
|
||||||
"""
|
"""
|
||||||
with override_settings(MODEL_FILE=Path("does_not_exist")):
|
|
||||||
Document.objects.create(
|
|
||||||
title="Test Document",
|
|
||||||
)
|
|
||||||
Tag.objects.create(name="Test Tag", matching_algorithm=Tag.MATCH_AUTO)
|
|
||||||
self.client.force_login(self.user)
|
self.client.force_login(self.user)
|
||||||
response = self.client.get(self.ENDPOINT)
|
response = self.client.get(self.ENDPOINT)
|
||||||
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
||||||
self.assertEqual(response.data["tasks"]["classifier_status"], "WARNING")
|
self.assertEqual(
|
||||||
self.assertIsNotNone(response.data["tasks"]["classifier_error"])
|
response.data["tasks"]["classifier_status"],
|
||||||
|
"WARNING",
|
||||||
@mock.patch(
|
|
||||||
"documents.classifier.load_classifier",
|
|
||||||
side_effect=ClassifierModelCorruptError(),
|
|
||||||
)
|
)
|
||||||
def test_system_status_classifier_error(self, mock_load_classifier):
|
|
||||||
|
def test_system_status_classifier_error(self):
|
||||||
"""
|
"""
|
||||||
GIVEN:
|
GIVEN:
|
||||||
- The classifier does exist but is corrupt
|
- An error occurred while loading the classifier
|
||||||
- > 0 documents and tags with auto matching exist
|
|
||||||
WHEN:
|
WHEN:
|
||||||
- The user requests the system status
|
- The user requests the system status
|
||||||
THEN:
|
THEN:
|
||||||
- The response contains an ERROR classifier status
|
- The response contains an ERROR classifier status
|
||||||
"""
|
"""
|
||||||
with (
|
PaperlessTask.objects.create(
|
||||||
tempfile.NamedTemporaryFile(
|
type=PaperlessTask.TaskType.SCHEDULED_TASK,
|
||||||
dir="/tmp",
|
status=states.FAILURE,
|
||||||
delete=False,
|
task_name=PaperlessTask.TaskName.TRAIN_CLASSIFIER,
|
||||||
) as does_exist,
|
result="Classifier training failed",
|
||||||
override_settings(MODEL_FILE=Path(does_exist.name)),
|
|
||||||
):
|
|
||||||
Document.objects.create(
|
|
||||||
title="Test Document",
|
|
||||||
)
|
|
||||||
Tag.objects.create(
|
|
||||||
name="Test Tag",
|
|
||||||
matching_algorithm=Tag.MATCH_AUTO,
|
|
||||||
)
|
)
|
||||||
self.client.force_login(self.user)
|
self.client.force_login(self.user)
|
||||||
response = self.client.get(self.ENDPOINT)
|
response = self.client.get(self.ENDPOINT)
|
||||||
@ -270,18 +250,63 @@ class TestSystemStatus(APITestCase):
|
|||||||
)
|
)
|
||||||
self.assertIsNotNone(response.data["tasks"]["classifier_error"])
|
self.assertIsNotNone(response.data["tasks"]["classifier_error"])
|
||||||
|
|
||||||
def test_system_status_classifier_ok_no_objects(self):
|
def test_system_status_sanity_check_ok(self):
|
||||||
"""
|
"""
|
||||||
GIVEN:
|
GIVEN:
|
||||||
- The classifier does not exist (and should not)
|
- The sanity check is successful
|
||||||
- No documents nor objects with auto matching exist
|
|
||||||
WHEN:
|
WHEN:
|
||||||
- The user requests the system status
|
- The user requests the system status
|
||||||
THEN:
|
THEN:
|
||||||
- The response contains an OK classifier status
|
- The response contains an OK sanity check status
|
||||||
"""
|
"""
|
||||||
with override_settings(MODEL_FILE=Path("does_not_exist")):
|
PaperlessTask.objects.create(
|
||||||
|
type=PaperlessTask.TaskType.SCHEDULED_TASK,
|
||||||
|
status=states.SUCCESS,
|
||||||
|
task_name=PaperlessTask.TaskName.CHECK_SANITY,
|
||||||
|
)
|
||||||
self.client.force_login(self.user)
|
self.client.force_login(self.user)
|
||||||
response = self.client.get(self.ENDPOINT)
|
response = self.client.get(self.ENDPOINT)
|
||||||
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
||||||
self.assertEqual(response.data["tasks"]["classifier_status"], "OK")
|
self.assertEqual(response.data["tasks"]["sanity_check_status"], "OK")
|
||||||
|
self.assertIsNone(response.data["tasks"]["sanity_check_error"])
|
||||||
|
|
||||||
|
def test_system_status_sanity_check_warning(self):
|
||||||
|
"""
|
||||||
|
GIVEN:
|
||||||
|
- No sanity check task is found
|
||||||
|
WHEN:
|
||||||
|
- The user requests the system status
|
||||||
|
THEN:
|
||||||
|
- The response contains a WARNING sanity check status
|
||||||
|
"""
|
||||||
|
self.client.force_login(self.user)
|
||||||
|
response = self.client.get(self.ENDPOINT)
|
||||||
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
||||||
|
self.assertEqual(
|
||||||
|
response.data["tasks"]["sanity_check_status"],
|
||||||
|
"WARNING",
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_system_status_sanity_check_error(self):
|
||||||
|
"""
|
||||||
|
GIVEN:
|
||||||
|
- The sanity check failed
|
||||||
|
WHEN:
|
||||||
|
- The user requests the system status
|
||||||
|
THEN:
|
||||||
|
- The response contains an ERROR sanity check status
|
||||||
|
"""
|
||||||
|
PaperlessTask.objects.create(
|
||||||
|
type=PaperlessTask.TaskType.SCHEDULED_TASK,
|
||||||
|
status=states.FAILURE,
|
||||||
|
task_name=PaperlessTask.TaskName.CHECK_SANITY,
|
||||||
|
result="5 issues found.",
|
||||||
|
)
|
||||||
|
self.client.force_login(self.user)
|
||||||
|
response = self.client.get(self.ENDPOINT)
|
||||||
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
||||||
|
self.assertEqual(
|
||||||
|
response.data["tasks"]["sanity_check_status"],
|
||||||
|
"ERROR",
|
||||||
|
)
|
||||||
|
self.assertIsNotNone(response.data["tasks"]["sanity_check_error"])
|
||||||
|
@ -130,7 +130,7 @@ class TestTasks(DirectoriesMixin, APITestCase):
|
|||||||
)
|
)
|
||||||
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
||||||
|
|
||||||
response = self.client.get(self.ENDPOINT)
|
response = self.client.get(self.ENDPOINT + "?acknowledged=false")
|
||||||
self.assertEqual(len(response.data), 0)
|
self.assertEqual(len(response.data), 0)
|
||||||
|
|
||||||
def test_tasks_owner_aware(self):
|
def test_tasks_owner_aware(self):
|
||||||
@ -246,7 +246,7 @@ class TestTasks(DirectoriesMixin, APITestCase):
|
|||||||
PaperlessTask.objects.create(
|
PaperlessTask.objects.create(
|
||||||
task_id=str(uuid.uuid4()),
|
task_id=str(uuid.uuid4()),
|
||||||
task_file_name="test.pdf",
|
task_file_name="test.pdf",
|
||||||
task_name="documents.tasks.some_task",
|
task_name=PaperlessTask.TaskName.CONSUME_FILE,
|
||||||
status=celery.states.SUCCESS,
|
status=celery.states.SUCCESS,
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -272,7 +272,7 @@ class TestTasks(DirectoriesMixin, APITestCase):
|
|||||||
PaperlessTask.objects.create(
|
PaperlessTask.objects.create(
|
||||||
task_id=str(uuid.uuid4()),
|
task_id=str(uuid.uuid4()),
|
||||||
task_file_name="anothertest.pdf",
|
task_file_name="anothertest.pdf",
|
||||||
task_name="documents.tasks.some_task",
|
task_name=PaperlessTask.TaskName.CONSUME_FILE,
|
||||||
status=celery.states.SUCCESS,
|
status=celery.states.SUCCESS,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -68,7 +68,7 @@ class TestTaskSignalHandler(DirectoriesMixin, TestCase):
|
|||||||
self.assertIsNotNone(task)
|
self.assertIsNotNone(task)
|
||||||
self.assertEqual(headers["id"], task.task_id)
|
self.assertEqual(headers["id"], task.task_id)
|
||||||
self.assertEqual("hello-999.pdf", task.task_file_name)
|
self.assertEqual("hello-999.pdf", task.task_file_name)
|
||||||
self.assertEqual("documents.tasks.consume_file", task.task_name)
|
self.assertEqual(PaperlessTask.TaskName.CONSUME_FILE, task.task_name)
|
||||||
self.assertEqual(1, task.owner_id)
|
self.assertEqual(1, task.owner_id)
|
||||||
self.assertEqual(celery.states.PENDING, task.status)
|
self.assertEqual(celery.states.PENDING, task.status)
|
||||||
|
|
||||||
|
@ -15,6 +15,7 @@ from urllib.parse import quote
|
|||||||
from urllib.parse import urlparse
|
from urllib.parse import urlparse
|
||||||
|
|
||||||
import pathvalidate
|
import pathvalidate
|
||||||
|
from celery import states
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.contrib.auth.models import Group
|
from django.contrib.auth.models import Group
|
||||||
from django.contrib.auth.models import User
|
from django.contrib.auth.models import User
|
||||||
@ -103,6 +104,7 @@ from documents.filters import DocumentsOrderingFilter
|
|||||||
from documents.filters import DocumentTypeFilterSet
|
from documents.filters import DocumentTypeFilterSet
|
||||||
from documents.filters import ObjectOwnedOrGrantedPermissionsFilter
|
from documents.filters import ObjectOwnedOrGrantedPermissionsFilter
|
||||||
from documents.filters import ObjectOwnedPermissionsFilter
|
from documents.filters import ObjectOwnedPermissionsFilter
|
||||||
|
from documents.filters import PaperlessTaskFilterSet
|
||||||
from documents.filters import ShareLinkFilterSet
|
from documents.filters import ShareLinkFilterSet
|
||||||
from documents.filters import StoragePathFilterSet
|
from documents.filters import StoragePathFilterSet
|
||||||
from documents.filters import TagFilterSet
|
from documents.filters import TagFilterSet
|
||||||
@ -2224,16 +2226,15 @@ class RemoteVersionView(GenericAPIView):
|
|||||||
class TasksViewSet(ReadOnlyModelViewSet):
|
class TasksViewSet(ReadOnlyModelViewSet):
|
||||||
permission_classes = (IsAuthenticated, PaperlessObjectPermissions)
|
permission_classes = (IsAuthenticated, PaperlessObjectPermissions)
|
||||||
serializer_class = TasksViewSerializer
|
serializer_class = TasksViewSerializer
|
||||||
filter_backends = (ObjectOwnedOrGrantedPermissionsFilter,)
|
filter_backends = (
|
||||||
|
DjangoFilterBackend,
|
||||||
|
OrderingFilter,
|
||||||
|
ObjectOwnedOrGrantedPermissionsFilter,
|
||||||
|
)
|
||||||
|
filterset_class = PaperlessTaskFilterSet
|
||||||
|
|
||||||
def get_queryset(self):
|
def get_queryset(self):
|
||||||
queryset = (
|
queryset = PaperlessTask.objects.all().order_by("-date_created")
|
||||||
PaperlessTask.objects.filter(
|
|
||||||
acknowledged=False,
|
|
||||||
)
|
|
||||||
.order_by("date_created")
|
|
||||||
.reverse()
|
|
||||||
)
|
|
||||||
task_id = self.request.query_params.get("task_id")
|
task_id = self.request.query_params.get("task_id")
|
||||||
if task_id is not None:
|
if task_id is not None:
|
||||||
queryset = PaperlessTask.objects.filter(task_id=task_id)
|
queryset = PaperlessTask.objects.filter(task_id=task_id)
|
||||||
@ -2562,6 +2563,14 @@ class CustomFieldViewSet(ModelViewSet):
|
|||||||
"last_trained": serializers.DateTimeField(),
|
"last_trained": serializers.DateTimeField(),
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
|
"sanity_check": inline_serializer(
|
||||||
|
name="SanityCheck",
|
||||||
|
fields={
|
||||||
|
"status": serializers.CharField(),
|
||||||
|
"error": serializers.CharField(),
|
||||||
|
"last_run": serializers.DateTimeField(),
|
||||||
|
},
|
||||||
|
),
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
},
|
},
|
||||||
@ -2570,6 +2579,17 @@ class CustomFieldViewSet(ModelViewSet):
|
|||||||
class SystemStatusView(PassUserMixin):
|
class SystemStatusView(PassUserMixin):
|
||||||
permission_classes = (IsAuthenticated,)
|
permission_classes = (IsAuthenticated,)
|
||||||
|
|
||||||
|
def _get_next_scheduled_task_schedule(
|
||||||
|
self,
|
||||||
|
schedule: dict,
|
||||||
|
task_name: str,
|
||||||
|
last_run,
|
||||||
|
) -> datetime | None:
|
||||||
|
# example: {'Check all e-mail accounts': {'task': 'paperless_mail.tasks.process_mail_accounts', 'schedule': <crontab: */10 * * * * (m/h/dM/MY/d)>, 'options': {'expires': 540.0}}, 'Train the classifier': {'task': 'documents.tasks.train_classifier', 'schedule': <crontab: 5 */1 * * * (m/h/dM/MY/d)>, 'options': {'expires': 3540.0}}, 'Optimize the index': {'task': 'documents.tasks.index_optimize', 'schedule': <crontab: 0 0 * * * (m/h/dM/MY/d)>, 'options': {'expires': 82800.0}}, 'Perform sanity check': {'task': 'documents.tasks.sanity_check', 'schedule': <crontab: 30 0 * * sun (m/h/dM/MY/d)>, 'options': {'expires': 601200.0}}, 'Empty trash': {'task': 'documents.tasks.empty_trash', 'schedule': <crontab: 0 1 * * * (m/h/dM/MY/d)>, 'options': {'expires': 82800.0}}, 'Check and run scheduled workflows': {'task': 'documents.tasks.check_scheduled_workflows', 'schedule': <crontab: 5 */1 * * * (m/h/dM/MY/d)>, 'options': {'expires': 3540.0}}}
|
||||||
|
for _, task_data in schedule.items():
|
||||||
|
if task_data["task"] and task_data["task"].find(task_name) != -1:
|
||||||
|
return task_data["schedule"]
|
||||||
|
|
||||||
def get(self, request, format=None):
|
def get(self, request, format=None):
|
||||||
if not request.user.is_staff:
|
if not request.user.is_staff:
|
||||||
return HttpResponseForbidden("Insufficient permissions")
|
return HttpResponseForbidden("Insufficient permissions")
|
||||||
@ -2622,13 +2642,22 @@ class SystemStatusView(PassUserMixin):
|
|||||||
)
|
)
|
||||||
redis_error = "Error connecting to redis, check logs for more detail."
|
redis_error = "Error connecting to redis, check logs for more detail."
|
||||||
|
|
||||||
|
celery_error = None
|
||||||
|
celery_url = None
|
||||||
|
schedule = None
|
||||||
try:
|
try:
|
||||||
celery_ping = celery_app.control.inspect().ping()
|
celery_ping = celery_app.control.inspect().ping()
|
||||||
first_worker_ping = celery_ping[next(iter(celery_ping.keys()))]
|
celery_url = next(iter(celery_ping.keys()))
|
||||||
|
first_worker_ping = celery_ping[celery_url]
|
||||||
|
schedule = celery_app.conf.beat_schedule
|
||||||
if first_worker_ping["ok"] == "pong":
|
if first_worker_ping["ok"] == "pong":
|
||||||
celery_active = "OK"
|
celery_active = "OK"
|
||||||
except Exception:
|
except Exception as e:
|
||||||
celery_active = "ERROR"
|
celery_active = "ERROR"
|
||||||
|
logger.exception(
|
||||||
|
f"System status detected a possible problem while connecting to celery: {e}",
|
||||||
|
)
|
||||||
|
celery_error = "Error connecting to celery, check logs for more detail."
|
||||||
|
|
||||||
index_error = None
|
index_error = None
|
||||||
try:
|
try:
|
||||||
@ -2645,54 +2674,72 @@ class SystemStatusView(PassUserMixin):
|
|||||||
)
|
)
|
||||||
index_last_modified = None
|
index_last_modified = None
|
||||||
|
|
||||||
classifier_error = None
|
last_trained_task = (
|
||||||
classifier_status = None
|
PaperlessTask.objects.filter(
|
||||||
try:
|
task_name=PaperlessTask.TaskName.TRAIN_CLASSIFIER,
|
||||||
classifier = load_classifier(raise_exception=True)
|
|
||||||
if classifier is None:
|
|
||||||
# Make sure classifier should exist
|
|
||||||
docs_queryset = Document.objects.exclude(
|
|
||||||
tags__is_inbox_tag=True,
|
|
||||||
)
|
)
|
||||||
if (
|
.order_by("-date_done")
|
||||||
docs_queryset.count() > 0
|
.first()
|
||||||
and (
|
|
||||||
Tag.objects.filter(matching_algorithm=Tag.MATCH_AUTO).exists()
|
|
||||||
or DocumentType.objects.filter(
|
|
||||||
matching_algorithm=Tag.MATCH_AUTO,
|
|
||||||
).exists()
|
|
||||||
or Correspondent.objects.filter(
|
|
||||||
matching_algorithm=Tag.MATCH_AUTO,
|
|
||||||
).exists()
|
|
||||||
or StoragePath.objects.filter(
|
|
||||||
matching_algorithm=Tag.MATCH_AUTO,
|
|
||||||
).exists()
|
|
||||||
)
|
)
|
||||||
and not settings.MODEL_FILE.exists()
|
|
||||||
):
|
|
||||||
# if classifier file doesn't exist just classify as a warning
|
|
||||||
classifier_error = "Classifier file does not exist (yet). Re-training may be pending."
|
|
||||||
classifier_status = "WARNING"
|
|
||||||
raise FileNotFoundError(classifier_error)
|
|
||||||
classifier_status = "OK"
|
classifier_status = "OK"
|
||||||
classifier_last_trained = (
|
classifier_error = None
|
||||||
make_aware(
|
classifier_next_training = None
|
||||||
datetime.fromtimestamp(classifier.get_last_checked()),
|
if last_trained_task is None:
|
||||||
)
|
classifier_status = "WARNING"
|
||||||
if settings.MODEL_FILE.exists()
|
classifier_error = "No classifier training tasks found"
|
||||||
and classifier.get_last_checked() is not None
|
elif last_trained_task and last_trained_task.status == states.FAILURE:
|
||||||
else None
|
|
||||||
)
|
|
||||||
except Exception as e:
|
|
||||||
if classifier_status is None:
|
|
||||||
classifier_status = "ERROR"
|
classifier_status = "ERROR"
|
||||||
classifier_last_trained = None
|
classifier_error = last_trained_task.result
|
||||||
if classifier_error is None:
|
classifier_last_trained = (
|
||||||
classifier_error = (
|
last_trained_task.date_done if last_trained_task else None
|
||||||
"Unable to load classifier, check logs for more detail."
|
|
||||||
)
|
)
|
||||||
logger.exception(
|
last_scheduled_trained_task = (
|
||||||
f"System status detected a possible problem while loading the classifier: {e}",
|
PaperlessTask.objects.filter(
|
||||||
|
task_name=PaperlessTask.TaskName.TRAIN_CLASSIFIER,
|
||||||
|
type=PaperlessTask.TaskType.SCHEDULED_TASK,
|
||||||
|
)
|
||||||
|
.order_by("-date_done")
|
||||||
|
.first()
|
||||||
|
)
|
||||||
|
if last_scheduled_trained_task and schedule:
|
||||||
|
classifier_next_training: datetime = self._get_next_scheduled_task_schedule(
|
||||||
|
schedule=schedule,
|
||||||
|
task_name=PaperlessTask.TaskName.TRAIN_CLASSIFIER,
|
||||||
|
last_run=last_trained_task.date_done,
|
||||||
|
)
|
||||||
|
|
||||||
|
last_sanity_check = (
|
||||||
|
PaperlessTask.objects.filter(
|
||||||
|
task_name=PaperlessTask.TaskName.CHECK_SANITY,
|
||||||
|
)
|
||||||
|
.order_by("-date_done")
|
||||||
|
.first()
|
||||||
|
)
|
||||||
|
sanity_check_status = "OK"
|
||||||
|
sanity_check_error = None
|
||||||
|
sanity_check_next_run = None
|
||||||
|
if last_sanity_check is None:
|
||||||
|
sanity_check_status = "WARNING"
|
||||||
|
sanity_check_error = "No sanity check tasks found"
|
||||||
|
elif last_sanity_check and last_sanity_check.status == states.FAILURE:
|
||||||
|
sanity_check_status = "ERROR"
|
||||||
|
sanity_check_error = last_sanity_check.result
|
||||||
|
sanity_check_last_run = (
|
||||||
|
last_sanity_check.date_done if last_sanity_check else None
|
||||||
|
)
|
||||||
|
last_scheduled_sanity_check = (
|
||||||
|
PaperlessTask.objects.filter(
|
||||||
|
task_name=PaperlessTask.TaskName.CHECK_SANITY,
|
||||||
|
type=PaperlessTask.TaskType.SCHEDULED_TASK,
|
||||||
|
)
|
||||||
|
.order_by("-date_done")
|
||||||
|
.first()
|
||||||
|
)
|
||||||
|
if last_scheduled_sanity_check and schedule:
|
||||||
|
sanity_check_next_run: datetime = self._get_next_scheduled_task_schedule(
|
||||||
|
schedule=schedule,
|
||||||
|
task_name=PaperlessTask.TaskName.CHECK_SANITY,
|
||||||
|
last_run=last_sanity_check.date_done,
|
||||||
)
|
)
|
||||||
|
|
||||||
return Response(
|
return Response(
|
||||||
@ -2721,12 +2768,19 @@ class SystemStatusView(PassUserMixin):
|
|||||||
"redis_status": redis_status,
|
"redis_status": redis_status,
|
||||||
"redis_error": redis_error,
|
"redis_error": redis_error,
|
||||||
"celery_status": celery_active,
|
"celery_status": celery_active,
|
||||||
|
"celery_url": celery_url,
|
||||||
|
"celery_error": celery_error,
|
||||||
"index_status": index_status,
|
"index_status": index_status,
|
||||||
"index_last_modified": index_last_modified,
|
"index_last_modified": index_last_modified,
|
||||||
"index_error": index_error,
|
"index_error": index_error,
|
||||||
"classifier_status": classifier_status,
|
"classifier_status": classifier_status,
|
||||||
"classifier_last_trained": classifier_last_trained,
|
"classifier_last_trained": classifier_last_trained,
|
||||||
|
"classifier_next_training": classifier_next_training,
|
||||||
"classifier_error": classifier_error,
|
"classifier_error": classifier_error,
|
||||||
|
"sanity_check_status": sanity_check_status,
|
||||||
|
"sanity_check_last_run": sanity_check_last_run,
|
||||||
|
"sanity_check_next_run": sanity_check_next_run,
|
||||||
|
"sanity_check_error": sanity_check_error,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
@ -2,7 +2,7 @@ msgid ""
|
|||||||
msgstr ""
|
msgstr ""
|
||||||
"Project-Id-Version: paperless-ngx\n"
|
"Project-Id-Version: paperless-ngx\n"
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"POT-Creation-Date: 2025-02-11 18:43-0800\n"
|
"POT-Creation-Date: 2025-02-14 15:45-0800\n"
|
||||||
"PO-Revision-Date: 2022-02-17 04:17\n"
|
"PO-Revision-Date: 2022-02-17 04:17\n"
|
||||||
"Last-Translator: \n"
|
"Last-Translator: \n"
|
||||||
"Language-Team: English\n"
|
"Language-Team: English\n"
|
||||||
@ -21,67 +21,67 @@ msgstr ""
|
|||||||
msgid "Documents"
|
msgid "Documents"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/filters.py:369
|
#: documents/filters.py:370
|
||||||
msgid "Value must be valid JSON."
|
msgid "Value must be valid JSON."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/filters.py:388
|
#: documents/filters.py:389
|
||||||
msgid "Invalid custom field query expression"
|
msgid "Invalid custom field query expression"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/filters.py:398
|
#: documents/filters.py:399
|
||||||
msgid "Invalid expression list. Must be nonempty."
|
msgid "Invalid expression list. Must be nonempty."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/filters.py:419
|
#: documents/filters.py:420
|
||||||
msgid "Invalid logical operator {op!r}"
|
msgid "Invalid logical operator {op!r}"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/filters.py:433
|
#: documents/filters.py:434
|
||||||
msgid "Maximum number of query conditions exceeded."
|
msgid "Maximum number of query conditions exceeded."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/filters.py:498
|
#: documents/filters.py:499
|
||||||
msgid "{name!r} is not a valid custom field."
|
msgid "{name!r} is not a valid custom field."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/filters.py:535
|
#: documents/filters.py:536
|
||||||
msgid "{data_type} does not support query expr {expr!r}."
|
msgid "{data_type} does not support query expr {expr!r}."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/filters.py:643
|
#: documents/filters.py:644
|
||||||
msgid "Maximum nesting depth exceeded."
|
msgid "Maximum nesting depth exceeded."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/filters.py:813
|
#: documents/filters.py:829
|
||||||
msgid "Custom field not found"
|
msgid "Custom field not found"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:41 documents/models.py:806
|
#: documents/models.py:41 documents/models.py:829
|
||||||
msgid "owner"
|
msgid "owner"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:58 documents/models.py:1017
|
#: documents/models.py:58 documents/models.py:1040
|
||||||
msgid "None"
|
msgid "None"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:59 documents/models.py:1018
|
#: documents/models.py:59 documents/models.py:1041
|
||||||
msgid "Any word"
|
msgid "Any word"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:60 documents/models.py:1019
|
#: documents/models.py:60 documents/models.py:1042
|
||||||
msgid "All words"
|
msgid "All words"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:61 documents/models.py:1020
|
#: documents/models.py:61 documents/models.py:1043
|
||||||
msgid "Exact match"
|
msgid "Exact match"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:62 documents/models.py:1021
|
#: documents/models.py:62 documents/models.py:1044
|
||||||
msgid "Regular expression"
|
msgid "Regular expression"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:63 documents/models.py:1022
|
#: documents/models.py:63 documents/models.py:1045
|
||||||
msgid "Fuzzy word"
|
msgid "Fuzzy word"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
@ -89,20 +89,20 @@ msgstr ""
|
|||||||
msgid "Automatic"
|
msgid "Automatic"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:67 documents/models.py:433 documents/models.py:1498
|
#: documents/models.py:67 documents/models.py:433 documents/models.py:1521
|
||||||
#: paperless_mail/models.py:23 paperless_mail/models.py:143
|
#: paperless_mail/models.py:23 paperless_mail/models.py:143
|
||||||
msgid "name"
|
msgid "name"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:69 documents/models.py:1085
|
#: documents/models.py:69 documents/models.py:1108
|
||||||
msgid "match"
|
msgid "match"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:72 documents/models.py:1088
|
#: documents/models.py:72 documents/models.py:1111
|
||||||
msgid "matching algorithm"
|
msgid "matching algorithm"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:77 documents/models.py:1093
|
#: documents/models.py:77 documents/models.py:1116
|
||||||
msgid "is insensitive"
|
msgid "is insensitive"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
@ -168,7 +168,7 @@ msgstr ""
|
|||||||
msgid "title"
|
msgid "title"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:175 documents/models.py:720
|
#: documents/models.py:175 documents/models.py:743
|
||||||
msgid "content"
|
msgid "content"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
@ -206,8 +206,8 @@ msgstr ""
|
|||||||
msgid "The number of pages of the document."
|
msgid "The number of pages of the document."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:221 documents/models.py:401 documents/models.py:726
|
#: documents/models.py:221 documents/models.py:401 documents/models.py:749
|
||||||
#: documents/models.py:764 documents/models.py:835 documents/models.py:893
|
#: documents/models.py:787 documents/models.py:858 documents/models.py:916
|
||||||
msgid "created"
|
msgid "created"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
@ -255,8 +255,8 @@ msgstr ""
|
|||||||
msgid "The position of this document in your physical document archive."
|
msgid "The position of this document in your physical document archive."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:295 documents/models.py:737 documents/models.py:791
|
#: documents/models.py:295 documents/models.py:760 documents/models.py:814
|
||||||
#: documents/models.py:1541
|
#: documents/models.py:1564
|
||||||
msgid "document"
|
msgid "document"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
@ -320,11 +320,11 @@ msgstr ""
|
|||||||
msgid "Title"
|
msgid "Title"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:420 documents/models.py:1037
|
#: documents/models.py:420 documents/models.py:1060
|
||||||
msgid "Created"
|
msgid "Created"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:421 documents/models.py:1036
|
#: documents/models.py:421 documents/models.py:1059
|
||||||
msgid "Added"
|
msgid "Added"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
@ -608,563 +608,595 @@ msgstr ""
|
|||||||
msgid "filter rules"
|
msgid "filter rules"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
|
#: documents/models.py:654
|
||||||
|
msgid "Auto Task"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: documents/models.py:655
|
||||||
|
msgid "Scheduled Task"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:656
|
#: documents/models.py:656
|
||||||
|
msgid "Manual Task"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: documents/models.py:659
|
||||||
|
msgid "Consume File"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: documents/models.py:660
|
||||||
|
msgid "Train Classifier"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: documents/models.py:661
|
||||||
|
msgid "Check Sanity"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: documents/models.py:666
|
||||||
msgid "Task ID"
|
msgid "Task ID"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:657
|
#: documents/models.py:667
|
||||||
msgid "Celery ID for the Task that was run"
|
msgid "Celery ID for the Task that was run"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:662
|
#: documents/models.py:672
|
||||||
msgid "Acknowledged"
|
msgid "Acknowledged"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:663
|
#: documents/models.py:673
|
||||||
msgid "If the task is acknowledged via the frontend or API"
|
msgid "If the task is acknowledged via the frontend or API"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:669
|
#: documents/models.py:679
|
||||||
msgid "Task Filename"
|
msgid "Task Filename"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:670
|
#: documents/models.py:680
|
||||||
msgid "Name of the file which the Task was run for"
|
msgid "Name of the file which the Task was run for"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:676
|
#: documents/models.py:687
|
||||||
msgid "Task Name"
|
msgid "Task Name"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:677
|
#: documents/models.py:688
|
||||||
msgid "Name of the Task which was run"
|
msgid "Name of the task that was run"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:684
|
#: documents/models.py:695
|
||||||
msgid "Task State"
|
msgid "Task State"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:685
|
#: documents/models.py:696
|
||||||
msgid "Current state of the task being run"
|
msgid "Current state of the task being run"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:690
|
#: documents/models.py:702
|
||||||
msgid "Created DateTime"
|
msgid "Created DateTime"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:691
|
#: documents/models.py:703
|
||||||
msgid "Datetime field when the task result was created in UTC"
|
msgid "Datetime field when the task result was created in UTC"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:696
|
#: documents/models.py:709
|
||||||
msgid "Started DateTime"
|
msgid "Started DateTime"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:697
|
#: documents/models.py:710
|
||||||
msgid "Datetime field when the task was started in UTC"
|
msgid "Datetime field when the task was started in UTC"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:702
|
#: documents/models.py:716
|
||||||
msgid "Completed DateTime"
|
msgid "Completed DateTime"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:703
|
#: documents/models.py:717
|
||||||
msgid "Datetime field when the task was completed in UTC"
|
msgid "Datetime field when the task was completed in UTC"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:708
|
#: documents/models.py:723
|
||||||
msgid "Result Data"
|
msgid "Result Data"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:710
|
#: documents/models.py:725
|
||||||
msgid "The data returned by the task"
|
msgid "The data returned by the task"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:722
|
#: documents/models.py:733
|
||||||
|
msgid "Task Type"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: documents/models.py:734
|
||||||
|
msgid "The type of task that was run"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: documents/models.py:745
|
||||||
msgid "Note for the document"
|
msgid "Note for the document"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:746
|
#: documents/models.py:769
|
||||||
msgid "user"
|
msgid "user"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:751
|
#: documents/models.py:774
|
||||||
msgid "note"
|
msgid "note"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:752
|
#: documents/models.py:775
|
||||||
msgid "notes"
|
msgid "notes"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:760
|
#: documents/models.py:783
|
||||||
msgid "Archive"
|
msgid "Archive"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:761
|
#: documents/models.py:784
|
||||||
msgid "Original"
|
msgid "Original"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:772 paperless_mail/models.py:75
|
#: documents/models.py:795 paperless_mail/models.py:75
|
||||||
msgid "expiration"
|
msgid "expiration"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:779
|
#: documents/models.py:802
|
||||||
msgid "slug"
|
msgid "slug"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:811
|
#: documents/models.py:834
|
||||||
msgid "share link"
|
msgid "share link"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:812
|
#: documents/models.py:835
|
||||||
msgid "share links"
|
msgid "share links"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:824
|
#: documents/models.py:847
|
||||||
msgid "String"
|
msgid "String"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:825
|
#: documents/models.py:848
|
||||||
msgid "URL"
|
msgid "URL"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:826
|
#: documents/models.py:849
|
||||||
msgid "Date"
|
msgid "Date"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:827
|
#: documents/models.py:850
|
||||||
msgid "Boolean"
|
msgid "Boolean"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:828
|
#: documents/models.py:851
|
||||||
msgid "Integer"
|
msgid "Integer"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:829
|
#: documents/models.py:852
|
||||||
msgid "Float"
|
msgid "Float"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:830
|
#: documents/models.py:853
|
||||||
msgid "Monetary"
|
msgid "Monetary"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:831
|
#: documents/models.py:854
|
||||||
msgid "Document Link"
|
msgid "Document Link"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:832
|
#: documents/models.py:855
|
||||||
msgid "Select"
|
msgid "Select"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:844
|
#: documents/models.py:867
|
||||||
msgid "data type"
|
msgid "data type"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:851
|
#: documents/models.py:874
|
||||||
msgid "extra data"
|
msgid "extra data"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:855
|
#: documents/models.py:878
|
||||||
msgid "Extra data for the custom field, such as select options"
|
msgid "Extra data for the custom field, such as select options"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:861
|
#: documents/models.py:884
|
||||||
msgid "custom field"
|
msgid "custom field"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:862
|
#: documents/models.py:885
|
||||||
msgid "custom fields"
|
msgid "custom fields"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:959
|
#: documents/models.py:982
|
||||||
msgid "custom field instance"
|
msgid "custom field instance"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:960
|
#: documents/models.py:983
|
||||||
msgid "custom field instances"
|
msgid "custom field instances"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1025
|
#: documents/models.py:1048
|
||||||
msgid "Consumption Started"
|
msgid "Consumption Started"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1026
|
#: documents/models.py:1049
|
||||||
msgid "Document Added"
|
msgid "Document Added"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1027
|
#: documents/models.py:1050
|
||||||
msgid "Document Updated"
|
msgid "Document Updated"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1028
|
#: documents/models.py:1051
|
||||||
msgid "Scheduled"
|
msgid "Scheduled"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1031
|
#: documents/models.py:1054
|
||||||
msgid "Consume Folder"
|
msgid "Consume Folder"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1032
|
#: documents/models.py:1055
|
||||||
msgid "Api Upload"
|
msgid "Api Upload"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1033
|
#: documents/models.py:1056
|
||||||
msgid "Mail Fetch"
|
msgid "Mail Fetch"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1038
|
#: documents/models.py:1061
|
||||||
msgid "Modified"
|
msgid "Modified"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1039
|
#: documents/models.py:1062
|
||||||
msgid "Custom Field"
|
msgid "Custom Field"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1042
|
#: documents/models.py:1065
|
||||||
msgid "Workflow Trigger Type"
|
msgid "Workflow Trigger Type"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1054
|
#: documents/models.py:1077
|
||||||
msgid "filter path"
|
msgid "filter path"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1059
|
#: documents/models.py:1082
|
||||||
msgid ""
|
msgid ""
|
||||||
"Only consume documents with a path that matches this if specified. Wildcards "
|
"Only consume documents with a path that matches this if specified. Wildcards "
|
||||||
"specified as * are allowed. Case insensitive."
|
"specified as * are allowed. Case insensitive."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1066
|
#: documents/models.py:1089
|
||||||
msgid "filter filename"
|
msgid "filter filename"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1071 paperless_mail/models.py:200
|
#: documents/models.py:1094 paperless_mail/models.py:200
|
||||||
msgid ""
|
msgid ""
|
||||||
"Only consume documents which entirely match this filename if specified. "
|
"Only consume documents which entirely match this filename if specified. "
|
||||||
"Wildcards such as *.pdf or *invoice* are allowed. Case insensitive."
|
"Wildcards such as *.pdf or *invoice* are allowed. Case insensitive."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1082
|
#: documents/models.py:1105
|
||||||
msgid "filter documents from this mail rule"
|
msgid "filter documents from this mail rule"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1098
|
#: documents/models.py:1121
|
||||||
msgid "has these tag(s)"
|
msgid "has these tag(s)"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1106
|
#: documents/models.py:1129
|
||||||
msgid "has this document type"
|
msgid "has this document type"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1114
|
#: documents/models.py:1137
|
||||||
msgid "has this correspondent"
|
msgid "has this correspondent"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1118
|
#: documents/models.py:1141
|
||||||
msgid "schedule offset days"
|
msgid "schedule offset days"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1121
|
#: documents/models.py:1144
|
||||||
msgid "The number of days to offset the schedule trigger by."
|
msgid "The number of days to offset the schedule trigger by."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1126
|
#: documents/models.py:1149
|
||||||
msgid "schedule is recurring"
|
msgid "schedule is recurring"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1129
|
#: documents/models.py:1152
|
||||||
msgid "If the schedule should be recurring."
|
msgid "If the schedule should be recurring."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1134
|
#: documents/models.py:1157
|
||||||
msgid "schedule recurring delay in days"
|
msgid "schedule recurring delay in days"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1138
|
#: documents/models.py:1161
|
||||||
msgid "The number of days between recurring schedule triggers."
|
msgid "The number of days between recurring schedule triggers."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1143
|
#: documents/models.py:1166
|
||||||
msgid "schedule date field"
|
msgid "schedule date field"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1148
|
#: documents/models.py:1171
|
||||||
msgid "The field to check for a schedule trigger."
|
msgid "The field to check for a schedule trigger."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1157
|
#: documents/models.py:1180
|
||||||
msgid "schedule date custom field"
|
msgid "schedule date custom field"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1161
|
#: documents/models.py:1184
|
||||||
msgid "workflow trigger"
|
msgid "workflow trigger"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1162
|
#: documents/models.py:1185
|
||||||
msgid "workflow triggers"
|
msgid "workflow triggers"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1170
|
#: documents/models.py:1193
|
||||||
msgid "email subject"
|
msgid "email subject"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1174
|
#: documents/models.py:1197
|
||||||
msgid ""
|
msgid ""
|
||||||
"The subject of the email, can include some placeholders, see documentation."
|
"The subject of the email, can include some placeholders, see documentation."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1180
|
#: documents/models.py:1203
|
||||||
msgid "email body"
|
msgid "email body"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1183
|
#: documents/models.py:1206
|
||||||
msgid ""
|
msgid ""
|
||||||
"The body (message) of the email, can include some placeholders, see "
|
"The body (message) of the email, can include some placeholders, see "
|
||||||
"documentation."
|
"documentation."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1189
|
#: documents/models.py:1212
|
||||||
msgid "emails to"
|
msgid "emails to"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1192
|
#: documents/models.py:1215
|
||||||
msgid "The destination email addresses, comma separated."
|
msgid "The destination email addresses, comma separated."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1198
|
#: documents/models.py:1221
|
||||||
msgid "include document in email"
|
msgid "include document in email"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1207
|
#: documents/models.py:1230
|
||||||
msgid "webhook url"
|
msgid "webhook url"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1209
|
#: documents/models.py:1232
|
||||||
msgid "The destination URL for the notification."
|
msgid "The destination URL for the notification."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1214
|
#: documents/models.py:1237
|
||||||
msgid "use parameters"
|
msgid "use parameters"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1219
|
#: documents/models.py:1242
|
||||||
msgid "send as JSON"
|
msgid "send as JSON"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1223
|
#: documents/models.py:1246
|
||||||
msgid "webhook parameters"
|
msgid "webhook parameters"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1226
|
#: documents/models.py:1249
|
||||||
msgid "The parameters to send with the webhook URL if body not used."
|
msgid "The parameters to send with the webhook URL if body not used."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1230
|
#: documents/models.py:1253
|
||||||
msgid "webhook body"
|
msgid "webhook body"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1233
|
#: documents/models.py:1256
|
||||||
msgid "The body to send with the webhook URL if parameters not used."
|
msgid "The body to send with the webhook URL if parameters not used."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1237
|
#: documents/models.py:1260
|
||||||
msgid "webhook headers"
|
msgid "webhook headers"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1240
|
#: documents/models.py:1263
|
||||||
msgid "The headers to send with the webhook URL."
|
msgid "The headers to send with the webhook URL."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1245
|
#: documents/models.py:1268
|
||||||
msgid "include document in webhook"
|
msgid "include document in webhook"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1256
|
#: documents/models.py:1279
|
||||||
msgid "Assignment"
|
msgid "Assignment"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1260
|
#: documents/models.py:1283
|
||||||
msgid "Removal"
|
msgid "Removal"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1264 documents/templates/account/password_reset.html:15
|
#: documents/models.py:1287 documents/templates/account/password_reset.html:15
|
||||||
msgid "Email"
|
msgid "Email"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1268
|
#: documents/models.py:1291
|
||||||
msgid "Webhook"
|
msgid "Webhook"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1272
|
#: documents/models.py:1295
|
||||||
msgid "Workflow Action Type"
|
msgid "Workflow Action Type"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1278
|
#: documents/models.py:1301
|
||||||
msgid "assign title"
|
msgid "assign title"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1283
|
#: documents/models.py:1306
|
||||||
msgid ""
|
msgid ""
|
||||||
"Assign a document title, can include some placeholders, see documentation."
|
"Assign a document title, can include some placeholders, see documentation."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1292 paperless_mail/models.py:274
|
#: documents/models.py:1315 paperless_mail/models.py:274
|
||||||
msgid "assign this tag"
|
msgid "assign this tag"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1301 paperless_mail/models.py:282
|
#: documents/models.py:1324 paperless_mail/models.py:282
|
||||||
msgid "assign this document type"
|
msgid "assign this document type"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1310 paperless_mail/models.py:296
|
#: documents/models.py:1333 paperless_mail/models.py:296
|
||||||
msgid "assign this correspondent"
|
msgid "assign this correspondent"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1319
|
#: documents/models.py:1342
|
||||||
msgid "assign this storage path"
|
msgid "assign this storage path"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1328
|
#: documents/models.py:1351
|
||||||
msgid "assign this owner"
|
msgid "assign this owner"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1335
|
#: documents/models.py:1358
|
||||||
msgid "grant view permissions to these users"
|
msgid "grant view permissions to these users"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1342
|
#: documents/models.py:1365
|
||||||
msgid "grant view permissions to these groups"
|
msgid "grant view permissions to these groups"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1349
|
#: documents/models.py:1372
|
||||||
msgid "grant change permissions to these users"
|
msgid "grant change permissions to these users"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1356
|
#: documents/models.py:1379
|
||||||
msgid "grant change permissions to these groups"
|
msgid "grant change permissions to these groups"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1363
|
#: documents/models.py:1386
|
||||||
msgid "assign these custom fields"
|
msgid "assign these custom fields"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1370
|
#: documents/models.py:1393
|
||||||
msgid "remove these tag(s)"
|
msgid "remove these tag(s)"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1375
|
#: documents/models.py:1398
|
||||||
msgid "remove all tags"
|
msgid "remove all tags"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1382
|
#: documents/models.py:1405
|
||||||
msgid "remove these document type(s)"
|
msgid "remove these document type(s)"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1387
|
#: documents/models.py:1410
|
||||||
msgid "remove all document types"
|
msgid "remove all document types"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1394
|
#: documents/models.py:1417
|
||||||
msgid "remove these correspondent(s)"
|
msgid "remove these correspondent(s)"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1399
|
#: documents/models.py:1422
|
||||||
msgid "remove all correspondents"
|
msgid "remove all correspondents"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1406
|
#: documents/models.py:1429
|
||||||
msgid "remove these storage path(s)"
|
msgid "remove these storage path(s)"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1411
|
#: documents/models.py:1434
|
||||||
msgid "remove all storage paths"
|
msgid "remove all storage paths"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1418
|
#: documents/models.py:1441
|
||||||
msgid "remove these owner(s)"
|
msgid "remove these owner(s)"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1423
|
#: documents/models.py:1446
|
||||||
msgid "remove all owners"
|
msgid "remove all owners"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1430
|
#: documents/models.py:1453
|
||||||
msgid "remove view permissions for these users"
|
msgid "remove view permissions for these users"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1437
|
#: documents/models.py:1460
|
||||||
msgid "remove view permissions for these groups"
|
msgid "remove view permissions for these groups"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1444
|
#: documents/models.py:1467
|
||||||
msgid "remove change permissions for these users"
|
msgid "remove change permissions for these users"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1451
|
#: documents/models.py:1474
|
||||||
msgid "remove change permissions for these groups"
|
msgid "remove change permissions for these groups"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1456
|
#: documents/models.py:1479
|
||||||
msgid "remove all permissions"
|
msgid "remove all permissions"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1463
|
#: documents/models.py:1486
|
||||||
msgid "remove these custom fields"
|
msgid "remove these custom fields"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1468
|
#: documents/models.py:1491
|
||||||
msgid "remove all custom fields"
|
msgid "remove all custom fields"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1477
|
#: documents/models.py:1500
|
||||||
msgid "email"
|
msgid "email"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1486
|
#: documents/models.py:1509
|
||||||
msgid "webhook"
|
msgid "webhook"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1490
|
#: documents/models.py:1513
|
||||||
msgid "workflow action"
|
msgid "workflow action"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1491
|
#: documents/models.py:1514
|
||||||
msgid "workflow actions"
|
msgid "workflow actions"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1500 paperless_mail/models.py:145
|
#: documents/models.py:1523 paperless_mail/models.py:145
|
||||||
msgid "order"
|
msgid "order"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1506
|
#: documents/models.py:1529
|
||||||
msgid "triggers"
|
msgid "triggers"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1513
|
#: documents/models.py:1536
|
||||||
msgid "actions"
|
msgid "actions"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1516 paperless_mail/models.py:154
|
#: documents/models.py:1539 paperless_mail/models.py:154
|
||||||
msgid "enabled"
|
msgid "enabled"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1527
|
#: documents/models.py:1550
|
||||||
msgid "workflow"
|
msgid "workflow"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1531
|
#: documents/models.py:1554
|
||||||
msgid "workflow trigger type"
|
msgid "workflow trigger type"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1545
|
#: documents/models.py:1568
|
||||||
msgid "date run"
|
msgid "date run"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1551
|
#: documents/models.py:1574
|
||||||
msgid "workflow run"
|
msgid "workflow run"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/models.py:1552
|
#: documents/models.py:1575
|
||||||
msgid "workflow runs"
|
msgid "workflow runs"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
@ -1402,21 +1434,6 @@ msgstr ""
|
|||||||
msgid "As a final step, please complete the following form:"
|
msgid "As a final step, please complete the following form:"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#: documents/validators.py:17
|
|
||||||
#, python-brace-format
|
|
||||||
msgid "Unable to parse URI {value}, missing scheme"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: documents/validators.py:22
|
|
||||||
#, python-brace-format
|
|
||||||
msgid "Unable to parse URI {value}, missing net location or path"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: documents/validators.py:27
|
|
||||||
#, python-brace-format
|
|
||||||
msgid "Unable to parse URI {value}"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: paperless/apps.py:10
|
#: paperless/apps.py:10
|
||||||
msgid "Paperless"
|
msgid "Paperless"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
Loading…
x
Reference in New Issue
Block a user