mirror of
https://github.com/paperless-ngx/paperless-ngx.git
synced 2025-04-02 13:45:10 -05:00
Compare commits
4 Commits
3339d7843f
...
1978209870
Author | SHA1 | Date | |
---|---|---|---|
![]() |
1978209870 | ||
![]() |
60796c770d | ||
![]() |
777eced98f | ||
![]() |
955ff32dcd |
@ -36,7 +36,13 @@ export const routes: Routes = [
|
|||||||
component: AppFrameComponent,
|
component: AppFrameComponent,
|
||||||
canDeactivate: [DirtyDocGuard],
|
canDeactivate: [DirtyDocGuard],
|
||||||
children: [
|
children: [
|
||||||
{ path: 'dashboard', component: DashboardComponent },
|
{
|
||||||
|
path: 'dashboard',
|
||||||
|
component: DashboardComponent,
|
||||||
|
data: {
|
||||||
|
componentName: 'AppFrameComponent',
|
||||||
|
},
|
||||||
|
},
|
||||||
{
|
{
|
||||||
path: 'documents',
|
path: 'documents',
|
||||||
component: DocumentListComponent,
|
component: DocumentListComponent,
|
||||||
@ -47,6 +53,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.View,
|
action: PermissionAction.View,
|
||||||
type: PermissionType.Document,
|
type: PermissionType.Document,
|
||||||
},
|
},
|
||||||
|
componentName: 'DocumentListComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -59,6 +66,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.View,
|
action: PermissionAction.View,
|
||||||
type: PermissionType.SavedView,
|
type: PermissionType.SavedView,
|
||||||
},
|
},
|
||||||
|
componentName: 'DocumentListComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -70,6 +78,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.View,
|
action: PermissionAction.View,
|
||||||
type: PermissionType.Document,
|
type: PermissionType.Document,
|
||||||
},
|
},
|
||||||
|
componentName: 'DocumentDetailComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -81,6 +90,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.View,
|
action: PermissionAction.View,
|
||||||
type: PermissionType.Document,
|
type: PermissionType.Document,
|
||||||
},
|
},
|
||||||
|
componentName: 'DocumentDetailComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -92,6 +102,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.View,
|
action: PermissionAction.View,
|
||||||
type: PermissionType.Document,
|
type: PermissionType.Document,
|
||||||
},
|
},
|
||||||
|
componentName: 'DocumentAsnComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -103,6 +114,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.View,
|
action: PermissionAction.View,
|
||||||
type: PermissionType.Tag,
|
type: PermissionType.Tag,
|
||||||
},
|
},
|
||||||
|
componentName: 'TagListComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -114,6 +126,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.View,
|
action: PermissionAction.View,
|
||||||
type: PermissionType.DocumentType,
|
type: PermissionType.DocumentType,
|
||||||
},
|
},
|
||||||
|
componentName: 'DocumentTypeListComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -125,6 +138,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.View,
|
action: PermissionAction.View,
|
||||||
type: PermissionType.Correspondent,
|
type: PermissionType.Correspondent,
|
||||||
},
|
},
|
||||||
|
componentName: 'CorrespondentListComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -136,6 +150,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.View,
|
action: PermissionAction.View,
|
||||||
type: PermissionType.StoragePath,
|
type: PermissionType.StoragePath,
|
||||||
},
|
},
|
||||||
|
componentName: 'StoragePathListComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -144,6 +159,7 @@ export const routes: Routes = [
|
|||||||
canActivate: [PermissionsGuard],
|
canActivate: [PermissionsGuard],
|
||||||
data: {
|
data: {
|
||||||
requireAdmin: true,
|
requireAdmin: true,
|
||||||
|
componentName: 'LogsComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -155,6 +171,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.Delete,
|
action: PermissionAction.Delete,
|
||||||
type: PermissionType.Document,
|
type: PermissionType.Document,
|
||||||
},
|
},
|
||||||
|
componentName: 'TrashComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
// redirect old paths
|
// redirect old paths
|
||||||
@ -180,6 +197,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.Change,
|
action: PermissionAction.Change,
|
||||||
type: PermissionType.UISettings,
|
type: PermissionType.UISettings,
|
||||||
},
|
},
|
||||||
|
componentName: 'SettingsComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -192,6 +210,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.View,
|
action: PermissionAction.View,
|
||||||
type: PermissionType.UISettings,
|
type: PermissionType.UISettings,
|
||||||
},
|
},
|
||||||
|
componentName: 'SettingsComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -203,6 +222,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.Change,
|
action: PermissionAction.Change,
|
||||||
type: PermissionType.AppConfig,
|
type: PermissionType.AppConfig,
|
||||||
},
|
},
|
||||||
|
componentName: 'ConfigComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -214,6 +234,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.View,
|
action: PermissionAction.View,
|
||||||
type: PermissionType.PaperlessTask,
|
type: PermissionType.PaperlessTask,
|
||||||
},
|
},
|
||||||
|
componentName: 'TasksComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -225,6 +246,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.View,
|
action: PermissionAction.View,
|
||||||
type: PermissionType.CustomField,
|
type: PermissionType.CustomField,
|
||||||
},
|
},
|
||||||
|
componentName: 'CustomFieldsComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -236,6 +258,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.View,
|
action: PermissionAction.View,
|
||||||
type: PermissionType.Workflow,
|
type: PermissionType.Workflow,
|
||||||
},
|
},
|
||||||
|
componentName: 'WorkflowsComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -247,6 +270,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.View,
|
action: PermissionAction.View,
|
||||||
type: PermissionType.MailAccount,
|
type: PermissionType.MailAccount,
|
||||||
},
|
},
|
||||||
|
componentName: 'MailComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -258,6 +282,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.View,
|
action: PermissionAction.View,
|
||||||
type: PermissionType.User,
|
type: PermissionType.User,
|
||||||
},
|
},
|
||||||
|
componentName: 'UsersAndGroupsComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -269,6 +294,7 @@ export const routes: Routes = [
|
|||||||
action: PermissionAction.View,
|
action: PermissionAction.View,
|
||||||
type: PermissionType.SavedView,
|
type: PermissionType.SavedView,
|
||||||
},
|
},
|
||||||
|
componentName: 'SavedViewsComponent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
|
@ -29,7 +29,7 @@ describe('ComponentRouterService', () => {
|
|||||||
eventsSubject.next(
|
eventsSubject.next(
|
||||||
new ActivationStart({
|
new ActivationStart({
|
||||||
url: 'test-url',
|
url: 'test-url',
|
||||||
component: { name: 'TestComponent' },
|
data: { componentName: 'TestComponent' },
|
||||||
} as any)
|
} as any)
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -41,13 +41,13 @@ describe('ComponentRouterService', () => {
|
|||||||
eventsSubject.next(
|
eventsSubject.next(
|
||||||
new ActivationStart({
|
new ActivationStart({
|
||||||
url: 'test-url-1',
|
url: 'test-url-1',
|
||||||
component: { name: 'TestComponent' },
|
data: { componentName: 'TestComponent' },
|
||||||
} as any)
|
} as any)
|
||||||
)
|
)
|
||||||
eventsSubject.next(
|
eventsSubject.next(
|
||||||
new ActivationStart({
|
new ActivationStart({
|
||||||
url: 'test-url-2',
|
url: 'test-url-2',
|
||||||
component: { name: 'TestComponent' },
|
data: { componentName: 'TestComponent' },
|
||||||
} as any)
|
} as any)
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -59,13 +59,13 @@ describe('ComponentRouterService', () => {
|
|||||||
eventsSubject.next(
|
eventsSubject.next(
|
||||||
new ActivationStart({
|
new ActivationStart({
|
||||||
url: 'test-url-1',
|
url: 'test-url-1',
|
||||||
component: { name: 'TestComponent1' },
|
data: { componentName: 'TestComponent1' },
|
||||||
} as any)
|
} as any)
|
||||||
)
|
)
|
||||||
eventsSubject.next(
|
eventsSubject.next(
|
||||||
new ActivationStart({
|
new ActivationStart({
|
||||||
url: 'test-url-2',
|
url: 'test-url-2',
|
||||||
component: { name: 'TestComponent2' },
|
data: { componentName: 'TestComponent2' },
|
||||||
} as any)
|
} as any)
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -76,13 +76,13 @@ describe('ComponentRouterService', () => {
|
|||||||
eventsSubject.next(
|
eventsSubject.next(
|
||||||
new ActivationStart({
|
new ActivationStart({
|
||||||
url: 'test-url-1',
|
url: 'test-url-1',
|
||||||
component: { name: 'TestComponent' },
|
data: { componentName: 'TestComponent' },
|
||||||
} as any)
|
} as any)
|
||||||
)
|
)
|
||||||
eventsSubject.next(
|
eventsSubject.next(
|
||||||
new ActivationStart({
|
new ActivationStart({
|
||||||
url: 'test-url-2',
|
url: 'test-url-2',
|
||||||
component: { name: 'TestComponent' },
|
data: { componentName: 'TestComponent' },
|
||||||
} as any)
|
} as any)
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -93,7 +93,7 @@ describe('ComponentRouterService', () => {
|
|||||||
eventsSubject.next(
|
eventsSubject.next(
|
||||||
new ActivationStart({
|
new ActivationStart({
|
||||||
url: 'test-url',
|
url: 'test-url',
|
||||||
component: { name: 'TestComponent' },
|
data: { componentName: 'TestComponent' },
|
||||||
} as any)
|
} as any)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -17,11 +17,11 @@ export class ComponentRouterService {
|
|||||||
.subscribe((event: ActivationStart) => {
|
.subscribe((event: ActivationStart) => {
|
||||||
if (
|
if (
|
||||||
this.componentHistory[this.componentHistory.length - 1] !==
|
this.componentHistory[this.componentHistory.length - 1] !==
|
||||||
event.snapshot.component.name &&
|
event.snapshot.data.componentName &&
|
||||||
!EXCLUDE_COMPONENTS.includes(event.snapshot.component.name)
|
!EXCLUDE_COMPONENTS.includes(event.snapshot.data.componentName)
|
||||||
) {
|
) {
|
||||||
this.history.push(event.snapshot.url.toString())
|
this.history.push(event.snapshot.url.toString())
|
||||||
this.componentHistory.push(event.snapshot.component.name)
|
this.componentHistory.push(event.snapshot.data.componentName)
|
||||||
} else {
|
} else {
|
||||||
// Update the URL of the current component in case the same component was loaded via a different URL
|
// Update the URL of the current component in case the same component was loaded via a different URL
|
||||||
this.history[this.history.length - 1] = event.snapshot.url.toString()
|
this.history[this.history.length - 1] = event.snapshot.url.toString()
|
||||||
|
@ -43,6 +43,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 MatchingModel
|
from documents.models import MatchingModel
|
||||||
|
from documents.models import Note
|
||||||
from documents.models import PaperlessTask
|
from documents.models import PaperlessTask
|
||||||
from documents.models import SavedView
|
from documents.models import SavedView
|
||||||
from documents.models import SavedViewFilterRule
|
from documents.models import SavedViewFilterRule
|
||||||
@ -861,6 +862,22 @@ class CustomFieldInstanceSerializer(serializers.ModelSerializer):
|
|||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class BasicUserSerializer(serializers.ModelSerializer):
|
||||||
|
# Different than paperless.serializers.UserSerializer
|
||||||
|
class Meta:
|
||||||
|
model = User
|
||||||
|
fields = ["id", "username", "first_name", "last_name"]
|
||||||
|
|
||||||
|
|
||||||
|
class NotesSerializer(serializers.ModelSerializer):
|
||||||
|
user = BasicUserSerializer()
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = Note
|
||||||
|
fields = ["id", "note", "created", "user"]
|
||||||
|
ordering = ["-created"]
|
||||||
|
|
||||||
|
|
||||||
class DocumentSerializer(
|
class DocumentSerializer(
|
||||||
OwnedObjectSerializer,
|
OwnedObjectSerializer,
|
||||||
NestedUpdateMixin,
|
NestedUpdateMixin,
|
||||||
@ -876,6 +893,8 @@ class DocumentSerializer(
|
|||||||
created_date = serializers.DateField(required=False)
|
created_date = serializers.DateField(required=False)
|
||||||
page_count = SerializerMethodField()
|
page_count = SerializerMethodField()
|
||||||
|
|
||||||
|
notes = NotesSerializer(many=True, required=False)
|
||||||
|
|
||||||
custom_fields = CustomFieldInstanceSerializer(
|
custom_fields = CustomFieldInstanceSerializer(
|
||||||
many=True,
|
many=True,
|
||||||
allow_null=False,
|
allow_null=False,
|
||||||
|
@ -2170,8 +2170,10 @@ class TestDocumentApi(DirectoriesMixin, DocumentConsumeDelayMixin, APITestCase):
|
|||||||
GIVEN:
|
GIVEN:
|
||||||
- A document with a single note
|
- A document with a single note
|
||||||
WHEN:
|
WHEN:
|
||||||
|
- API request for document
|
||||||
- API request for document notes is made
|
- API request for document notes is made
|
||||||
THEN:
|
THEN:
|
||||||
|
- Note is included in the document response
|
||||||
- The associated note is returned
|
- The associated note is returned
|
||||||
"""
|
"""
|
||||||
doc = Document.objects.create(
|
doc = Document.objects.create(
|
||||||
@ -2185,6 +2187,18 @@ class TestDocumentApi(DirectoriesMixin, DocumentConsumeDelayMixin, APITestCase):
|
|||||||
user=self.user,
|
user=self.user,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
response = self.client.get(
|
||||||
|
f"/api/documents/{doc.pk}/",
|
||||||
|
format="json",
|
||||||
|
)
|
||||||
|
|
||||||
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
||||||
|
|
||||||
|
resp_data = response.json()
|
||||||
|
self.assertEqual(len(resp_data["notes"]), 1)
|
||||||
|
self.assertEqual(resp_data["notes"][0]["note"], note.note)
|
||||||
|
self.assertEqual(resp_data["notes"][0]["user"]["username"], self.user.username)
|
||||||
|
|
||||||
response = self.client.get(
|
response = self.client.get(
|
||||||
f"/api/documents/{doc.pk}/notes/",
|
f"/api/documents/{doc.pk}/notes/",
|
||||||
format="json",
|
format="json",
|
||||||
|
@ -803,33 +803,6 @@ class DocumentViewSet(
|
|||||||
except (FileNotFoundError, Document.DoesNotExist):
|
except (FileNotFoundError, Document.DoesNotExist):
|
||||||
raise Http404
|
raise Http404
|
||||||
|
|
||||||
def getNotes(self, doc):
|
|
||||||
return [
|
|
||||||
{
|
|
||||||
"id": c.pk,
|
|
||||||
"note": c.note,
|
|
||||||
"created": c.created,
|
|
||||||
"user": {
|
|
||||||
"id": c.user.id,
|
|
||||||
"username": c.user.username,
|
|
||||||
"first_name": c.user.first_name,
|
|
||||||
"last_name": c.user.last_name,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
for c in Note.objects.select_related("user")
|
|
||||||
.only(
|
|
||||||
"pk",
|
|
||||||
"note",
|
|
||||||
"created",
|
|
||||||
"user__id",
|
|
||||||
"user__username",
|
|
||||||
"user__first_name",
|
|
||||||
"user__last_name",
|
|
||||||
)
|
|
||||||
.filter(document=doc)
|
|
||||||
.order_by("-created")
|
|
||||||
]
|
|
||||||
|
|
||||||
@action(
|
@action(
|
||||||
methods=["get", "post", "delete"],
|
methods=["get", "post", "delete"],
|
||||||
detail=True,
|
detail=True,
|
||||||
@ -854,9 +827,11 @@ class DocumentViewSet(
|
|||||||
except Document.DoesNotExist:
|
except Document.DoesNotExist:
|
||||||
raise Http404
|
raise Http404
|
||||||
|
|
||||||
|
serializer = self.get_serializer(doc)
|
||||||
|
|
||||||
if request.method == "GET":
|
if request.method == "GET":
|
||||||
try:
|
try:
|
||||||
notes = self.getNotes(doc)
|
notes = serializer.to_representation(doc).get("notes")
|
||||||
return Response(notes)
|
return Response(notes)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logger.warning(f"An error occurred retrieving notes: {e!s}")
|
logger.warning(f"An error occurred retrieving notes: {e!s}")
|
||||||
@ -897,7 +872,7 @@ class DocumentViewSet(
|
|||||||
|
|
||||||
index.add_or_update_document(doc)
|
index.add_or_update_document(doc)
|
||||||
|
|
||||||
notes = self.getNotes(doc)
|
notes = serializer.to_representation(doc).get("notes")
|
||||||
|
|
||||||
return Response(notes)
|
return Response(notes)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
@ -934,7 +909,9 @@ class DocumentViewSet(
|
|||||||
|
|
||||||
index.add_or_update_document(doc)
|
index.add_or_update_document(doc)
|
||||||
|
|
||||||
return Response(self.getNotes(doc))
|
notes = serializer.to_representation(doc).get("notes")
|
||||||
|
|
||||||
|
return Response(notes)
|
||||||
|
|
||||||
return Response(
|
return Response(
|
||||||
{
|
{
|
||||||
|
Loading…
x
Reference in New Issue
Block a user