mirror of
				https://github.com/paperless-ngx/paperless-ngx.git
				synced 2025-10-25 15:52:35 -04:00 
			
		
		
		
	Chore: reorganize api tests (#4935)
* Move permissions-related API tests * Move bulk-edit-related API tests * Move bulk-download-related API tests * Move uisettings-related API tests * Move remoteversion-related API tests * Move tasks API tests * Move object-related API tests * Move consumption-template-related API tests * Rename pared-down documents API test file Co-Authored-By: Trenton H <797416+stumpylog@users.noreply.github.com>
This commit is contained in:
		
							parent
							
								
									85f824f032
								
							
						
					
					
						commit
						e2d25a7a09
					
				
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										337
									
								
								src/documents/tests/test_api_bulk_download.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										337
									
								
								src/documents/tests/test_api_bulk_download.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,337 @@ | ||||
| import datetime | ||||
| import io | ||||
| import json | ||||
| import os | ||||
| import shutil | ||||
| import zipfile | ||||
| 
 | ||||
| from django.contrib.auth.models import User | ||||
| from django.test import override_settings | ||||
| from django.utils import timezone | ||||
| from rest_framework import status | ||||
| from rest_framework.test import APITestCase | ||||
| 
 | ||||
| from documents.models import Correspondent | ||||
| from documents.models import Document | ||||
| from documents.models import DocumentType | ||||
| from documents.tests.utils import DirectoriesMixin | ||||
| 
 | ||||
| 
 | ||||
| class TestBulkDownload(DirectoriesMixin, APITestCase): | ||||
|     ENDPOINT = "/api/documents/bulk_download/" | ||||
| 
 | ||||
|     def setUp(self): | ||||
|         super().setUp() | ||||
| 
 | ||||
|         user = User.objects.create_superuser(username="temp_admin") | ||||
|         self.client.force_authenticate(user=user) | ||||
| 
 | ||||
|         self.doc1 = Document.objects.create(title="unrelated", checksum="A") | ||||
|         self.doc2 = Document.objects.create( | ||||
|             title="document A", | ||||
|             filename="docA.pdf", | ||||
|             mime_type="application/pdf", | ||||
|             checksum="B", | ||||
|             created=timezone.make_aware(datetime.datetime(2021, 1, 1)), | ||||
|         ) | ||||
|         self.doc2b = Document.objects.create( | ||||
|             title="document A", | ||||
|             filename="docA2.pdf", | ||||
|             mime_type="application/pdf", | ||||
|             checksum="D", | ||||
|             created=timezone.make_aware(datetime.datetime(2021, 1, 1)), | ||||
|         ) | ||||
|         self.doc3 = Document.objects.create( | ||||
|             title="document B", | ||||
|             filename="docB.jpg", | ||||
|             mime_type="image/jpeg", | ||||
|             checksum="C", | ||||
|             created=timezone.make_aware(datetime.datetime(2020, 3, 21)), | ||||
|             archive_filename="docB.pdf", | ||||
|             archive_checksum="D", | ||||
|         ) | ||||
| 
 | ||||
|         shutil.copy( | ||||
|             os.path.join(os.path.dirname(__file__), "samples", "simple.pdf"), | ||||
|             self.doc2.source_path, | ||||
|         ) | ||||
|         shutil.copy( | ||||
|             os.path.join(os.path.dirname(__file__), "samples", "simple.png"), | ||||
|             self.doc2b.source_path, | ||||
|         ) | ||||
|         shutil.copy( | ||||
|             os.path.join(os.path.dirname(__file__), "samples", "simple.jpg"), | ||||
|             self.doc3.source_path, | ||||
|         ) | ||||
|         shutil.copy( | ||||
|             os.path.join(os.path.dirname(__file__), "samples", "test_with_bom.pdf"), | ||||
|             self.doc3.archive_path, | ||||
|         ) | ||||
| 
 | ||||
|     def test_download_originals(self): | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             json.dumps( | ||||
|                 {"documents": [self.doc2.id, self.doc3.id], "content": "originals"}, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(response["Content-Type"], "application/zip") | ||||
| 
 | ||||
|         with zipfile.ZipFile(io.BytesIO(response.content)) as zipf: | ||||
|             self.assertEqual(len(zipf.filelist), 2) | ||||
|             self.assertIn("2021-01-01 document A.pdf", zipf.namelist()) | ||||
|             self.assertIn("2020-03-21 document B.jpg", zipf.namelist()) | ||||
| 
 | ||||
|             with self.doc2.source_file as f: | ||||
|                 self.assertEqual(f.read(), zipf.read("2021-01-01 document A.pdf")) | ||||
| 
 | ||||
|             with self.doc3.source_file as f: | ||||
|                 self.assertEqual(f.read(), zipf.read("2020-03-21 document B.jpg")) | ||||
| 
 | ||||
|     def test_download_default(self): | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             json.dumps({"documents": [self.doc2.id, self.doc3.id]}), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(response["Content-Type"], "application/zip") | ||||
| 
 | ||||
|         with zipfile.ZipFile(io.BytesIO(response.content)) as zipf: | ||||
|             self.assertEqual(len(zipf.filelist), 2) | ||||
|             self.assertIn("2021-01-01 document A.pdf", zipf.namelist()) | ||||
|             self.assertIn("2020-03-21 document B.pdf", zipf.namelist()) | ||||
| 
 | ||||
|             with self.doc2.source_file as f: | ||||
|                 self.assertEqual(f.read(), zipf.read("2021-01-01 document A.pdf")) | ||||
| 
 | ||||
|             with self.doc3.archive_file as f: | ||||
|                 self.assertEqual(f.read(), zipf.read("2020-03-21 document B.pdf")) | ||||
| 
 | ||||
|     def test_download_both(self): | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             json.dumps({"documents": [self.doc2.id, self.doc3.id], "content": "both"}), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(response["Content-Type"], "application/zip") | ||||
| 
 | ||||
|         with zipfile.ZipFile(io.BytesIO(response.content)) as zipf: | ||||
|             self.assertEqual(len(zipf.filelist), 3) | ||||
|             self.assertIn("originals/2021-01-01 document A.pdf", zipf.namelist()) | ||||
|             self.assertIn("archive/2020-03-21 document B.pdf", zipf.namelist()) | ||||
|             self.assertIn("originals/2020-03-21 document B.jpg", zipf.namelist()) | ||||
| 
 | ||||
|             with self.doc2.source_file as f: | ||||
|                 self.assertEqual( | ||||
|                     f.read(), | ||||
|                     zipf.read("originals/2021-01-01 document A.pdf"), | ||||
|                 ) | ||||
| 
 | ||||
|             with self.doc3.archive_file as f: | ||||
|                 self.assertEqual( | ||||
|                     f.read(), | ||||
|                     zipf.read("archive/2020-03-21 document B.pdf"), | ||||
|                 ) | ||||
| 
 | ||||
|             with self.doc3.source_file as f: | ||||
|                 self.assertEqual( | ||||
|                     f.read(), | ||||
|                     zipf.read("originals/2020-03-21 document B.jpg"), | ||||
|                 ) | ||||
| 
 | ||||
|     def test_filename_clashes(self): | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             json.dumps({"documents": [self.doc2.id, self.doc2b.id]}), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(response["Content-Type"], "application/zip") | ||||
| 
 | ||||
|         with zipfile.ZipFile(io.BytesIO(response.content)) as zipf: | ||||
|             self.assertEqual(len(zipf.filelist), 2) | ||||
| 
 | ||||
|             self.assertIn("2021-01-01 document A.pdf", zipf.namelist()) | ||||
|             self.assertIn("2021-01-01 document A_01.pdf", zipf.namelist()) | ||||
| 
 | ||||
|             with self.doc2.source_file as f: | ||||
|                 self.assertEqual(f.read(), zipf.read("2021-01-01 document A.pdf")) | ||||
| 
 | ||||
|             with self.doc2b.source_file as f: | ||||
|                 self.assertEqual(f.read(), zipf.read("2021-01-01 document A_01.pdf")) | ||||
| 
 | ||||
|     def test_compression(self): | ||||
|         self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             json.dumps( | ||||
|                 {"documents": [self.doc2.id, self.doc2b.id], "compression": "lzma"}, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|     @override_settings(FILENAME_FORMAT="{correspondent}/{title}") | ||||
|     def test_formatted_download_originals(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Defined file naming format | ||||
|         WHEN: | ||||
|             - Bulk download request for original documents | ||||
|             - Bulk download request requests to follow format | ||||
|         THEN: | ||||
|             - Files in resulting zipfile are formatted | ||||
|         """ | ||||
| 
 | ||||
|         c = Correspondent.objects.create(name="test") | ||||
|         c2 = Correspondent.objects.create(name="a space name") | ||||
| 
 | ||||
|         self.doc2.correspondent = c | ||||
|         self.doc2.title = "This is Doc 2" | ||||
|         self.doc2.save() | ||||
| 
 | ||||
|         self.doc3.correspondent = c2 | ||||
|         self.doc3.title = "Title 2 - Doc 3" | ||||
|         self.doc3.save() | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc2.id, self.doc3.id], | ||||
|                     "content": "originals", | ||||
|                     "follow_formatting": True, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(response["Content-Type"], "application/zip") | ||||
| 
 | ||||
|         with zipfile.ZipFile(io.BytesIO(response.content)) as zipf: | ||||
|             self.assertEqual(len(zipf.filelist), 2) | ||||
|             self.assertIn("a space name/Title 2 - Doc 3.jpg", zipf.namelist()) | ||||
|             self.assertIn("test/This is Doc 2.pdf", zipf.namelist()) | ||||
| 
 | ||||
|             with self.doc2.source_file as f: | ||||
|                 self.assertEqual(f.read(), zipf.read("test/This is Doc 2.pdf")) | ||||
| 
 | ||||
|             with self.doc3.source_file as f: | ||||
|                 self.assertEqual( | ||||
|                     f.read(), | ||||
|                     zipf.read("a space name/Title 2 - Doc 3.jpg"), | ||||
|                 ) | ||||
| 
 | ||||
|     @override_settings(FILENAME_FORMAT="somewhere/{title}") | ||||
|     def test_formatted_download_archive(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Defined file naming format | ||||
|         WHEN: | ||||
|             - Bulk download request for archive documents | ||||
|             - Bulk download request requests to follow format | ||||
|         THEN: | ||||
|             - Files in resulting zipfile are formatted | ||||
|         """ | ||||
| 
 | ||||
|         self.doc2.title = "This is Doc 2" | ||||
|         self.doc2.save() | ||||
| 
 | ||||
|         self.doc3.title = "Title 2 - Doc 3" | ||||
|         self.doc3.save() | ||||
|         print(self.doc3.archive_path) | ||||
|         print(self.doc3.archive_filename) | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc2.id, self.doc3.id], | ||||
|                     "follow_formatting": True, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(response["Content-Type"], "application/zip") | ||||
| 
 | ||||
|         with zipfile.ZipFile(io.BytesIO(response.content)) as zipf: | ||||
|             self.assertEqual(len(zipf.filelist), 2) | ||||
|             self.assertIn("somewhere/This is Doc 2.pdf", zipf.namelist()) | ||||
|             self.assertIn("somewhere/Title 2 - Doc 3.pdf", zipf.namelist()) | ||||
| 
 | ||||
|             with self.doc2.source_file as f: | ||||
|                 self.assertEqual(f.read(), zipf.read("somewhere/This is Doc 2.pdf")) | ||||
| 
 | ||||
|             with self.doc3.archive_file as f: | ||||
|                 self.assertEqual(f.read(), zipf.read("somewhere/Title 2 - Doc 3.pdf")) | ||||
| 
 | ||||
|     @override_settings(FILENAME_FORMAT="{document_type}/{title}") | ||||
|     def test_formatted_download_both(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Defined file naming format | ||||
|         WHEN: | ||||
|             - Bulk download request for original documents and archive documents | ||||
|             - Bulk download request requests to follow format | ||||
|         THEN: | ||||
|             - Files defined in resulting zipfile are formatted | ||||
|         """ | ||||
| 
 | ||||
|         dc1 = DocumentType.objects.create(name="bill") | ||||
|         dc2 = DocumentType.objects.create(name="statement") | ||||
| 
 | ||||
|         self.doc2.document_type = dc1 | ||||
|         self.doc2.title = "This is Doc 2" | ||||
|         self.doc2.save() | ||||
| 
 | ||||
|         self.doc3.document_type = dc2 | ||||
|         self.doc3.title = "Title 2 - Doc 3" | ||||
|         self.doc3.save() | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc2.id, self.doc3.id], | ||||
|                     "content": "both", | ||||
|                     "follow_formatting": True, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(response["Content-Type"], "application/zip") | ||||
| 
 | ||||
|         with zipfile.ZipFile(io.BytesIO(response.content)) as zipf: | ||||
|             self.assertEqual(len(zipf.filelist), 3) | ||||
|             self.assertIn("originals/bill/This is Doc 2.pdf", zipf.namelist()) | ||||
|             self.assertIn("archive/statement/Title 2 - Doc 3.pdf", zipf.namelist()) | ||||
|             self.assertIn("originals/statement/Title 2 - Doc 3.jpg", zipf.namelist()) | ||||
| 
 | ||||
|             with self.doc2.source_file as f: | ||||
|                 self.assertEqual( | ||||
|                     f.read(), | ||||
|                     zipf.read("originals/bill/This is Doc 2.pdf"), | ||||
|                 ) | ||||
| 
 | ||||
|             with self.doc3.archive_file as f: | ||||
|                 self.assertEqual( | ||||
|                     f.read(), | ||||
|                     zipf.read("archive/statement/Title 2 - Doc 3.pdf"), | ||||
|                 ) | ||||
| 
 | ||||
|             with self.doc3.source_file as f: | ||||
|                 self.assertEqual( | ||||
|                     f.read(), | ||||
|                     zipf.read("originals/statement/Title 2 - Doc 3.jpg"), | ||||
|                 ) | ||||
							
								
								
									
										870
									
								
								src/documents/tests/test_api_bulk_edit.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										870
									
								
								src/documents/tests/test_api_bulk_edit.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,870 @@ | ||||
| import json | ||||
| from unittest import mock | ||||
| 
 | ||||
| from django.contrib.auth.models import User | ||||
| from guardian.shortcuts import assign_perm | ||||
| from rest_framework import status | ||||
| from rest_framework.test import APITestCase | ||||
| 
 | ||||
| from documents import bulk_edit | ||||
| from documents.models import Correspondent | ||||
| from documents.models import Document | ||||
| from documents.models import DocumentType | ||||
| from documents.models import StoragePath | ||||
| from documents.models import Tag | ||||
| from documents.tests.utils import DirectoriesMixin | ||||
| 
 | ||||
| 
 | ||||
| class TestBulkEdit(DirectoriesMixin, APITestCase): | ||||
|     def setUp(self): | ||||
|         super().setUp() | ||||
| 
 | ||||
|         user = User.objects.create_superuser(username="temp_admin") | ||||
|         self.client.force_authenticate(user=user) | ||||
| 
 | ||||
|         patcher = mock.patch("documents.bulk_edit.bulk_update_documents.delay") | ||||
|         self.async_task = patcher.start() | ||||
|         self.addCleanup(patcher.stop) | ||||
|         self.c1 = Correspondent.objects.create(name="c1") | ||||
|         self.c2 = Correspondent.objects.create(name="c2") | ||||
|         self.dt1 = DocumentType.objects.create(name="dt1") | ||||
|         self.dt2 = DocumentType.objects.create(name="dt2") | ||||
|         self.t1 = Tag.objects.create(name="t1") | ||||
|         self.t2 = Tag.objects.create(name="t2") | ||||
|         self.doc1 = Document.objects.create(checksum="A", title="A") | ||||
|         self.doc2 = Document.objects.create( | ||||
|             checksum="B", | ||||
|             title="B", | ||||
|             correspondent=self.c1, | ||||
|             document_type=self.dt1, | ||||
|         ) | ||||
|         self.doc3 = Document.objects.create( | ||||
|             checksum="C", | ||||
|             title="C", | ||||
|             correspondent=self.c2, | ||||
|             document_type=self.dt2, | ||||
|         ) | ||||
|         self.doc4 = Document.objects.create(checksum="D", title="D") | ||||
|         self.doc5 = Document.objects.create(checksum="E", title="E") | ||||
|         self.doc2.tags.add(self.t1) | ||||
|         self.doc3.tags.add(self.t2) | ||||
|         self.doc4.tags.add(self.t1, self.t2) | ||||
|         self.sp1 = StoragePath.objects.create(name="sp1", path="Something/{checksum}") | ||||
| 
 | ||||
|     def test_set_correspondent(self): | ||||
|         self.assertEqual(Document.objects.filter(correspondent=self.c2).count(), 1) | ||||
|         bulk_edit.set_correspondent( | ||||
|             [self.doc1.id, self.doc2.id, self.doc3.id], | ||||
|             self.c2.id, | ||||
|         ) | ||||
|         self.assertEqual(Document.objects.filter(correspondent=self.c2).count(), 3) | ||||
|         self.async_task.assert_called_once() | ||||
|         args, kwargs = self.async_task.call_args | ||||
|         self.assertCountEqual(kwargs["document_ids"], [self.doc1.id, self.doc2.id]) | ||||
| 
 | ||||
|     def test_unset_correspondent(self): | ||||
|         self.assertEqual(Document.objects.filter(correspondent=self.c2).count(), 1) | ||||
|         bulk_edit.set_correspondent([self.doc1.id, self.doc2.id, self.doc3.id], None) | ||||
|         self.assertEqual(Document.objects.filter(correspondent=self.c2).count(), 0) | ||||
|         self.async_task.assert_called_once() | ||||
|         args, kwargs = self.async_task.call_args | ||||
|         self.assertCountEqual(kwargs["document_ids"], [self.doc2.id, self.doc3.id]) | ||||
| 
 | ||||
|     def test_set_document_type(self): | ||||
|         self.assertEqual(Document.objects.filter(document_type=self.dt2).count(), 1) | ||||
|         bulk_edit.set_document_type( | ||||
|             [self.doc1.id, self.doc2.id, self.doc3.id], | ||||
|             self.dt2.id, | ||||
|         ) | ||||
|         self.assertEqual(Document.objects.filter(document_type=self.dt2).count(), 3) | ||||
|         self.async_task.assert_called_once() | ||||
|         args, kwargs = self.async_task.call_args | ||||
|         self.assertCountEqual(kwargs["document_ids"], [self.doc1.id, self.doc2.id]) | ||||
| 
 | ||||
|     def test_unset_document_type(self): | ||||
|         self.assertEqual(Document.objects.filter(document_type=self.dt2).count(), 1) | ||||
|         bulk_edit.set_document_type([self.doc1.id, self.doc2.id, self.doc3.id], None) | ||||
|         self.assertEqual(Document.objects.filter(document_type=self.dt2).count(), 0) | ||||
|         self.async_task.assert_called_once() | ||||
|         args, kwargs = self.async_task.call_args | ||||
|         self.assertCountEqual(kwargs["document_ids"], [self.doc2.id, self.doc3.id]) | ||||
| 
 | ||||
|     def test_set_document_storage_path(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - 5 documents without defined storage path | ||||
|         WHEN: | ||||
|             - Bulk edit called to add storage path to 1 document | ||||
|         THEN: | ||||
|             - Single document storage path update | ||||
|         """ | ||||
|         self.assertEqual(Document.objects.filter(storage_path=None).count(), 5) | ||||
| 
 | ||||
|         bulk_edit.set_storage_path( | ||||
|             [self.doc1.id], | ||||
|             self.sp1.id, | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(Document.objects.filter(storage_path=None).count(), 4) | ||||
| 
 | ||||
|         self.async_task.assert_called_once() | ||||
|         args, kwargs = self.async_task.call_args | ||||
| 
 | ||||
|         self.assertCountEqual(kwargs["document_ids"], [self.doc1.id]) | ||||
| 
 | ||||
|     def test_unset_document_storage_path(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - 4 documents without defined storage path | ||||
|             - 1 document with a defined storage | ||||
|         WHEN: | ||||
|             - Bulk edit called to remove storage path from 1 document | ||||
|         THEN: | ||||
|             - Single document storage path removed | ||||
|         """ | ||||
|         self.assertEqual(Document.objects.filter(storage_path=None).count(), 5) | ||||
| 
 | ||||
|         bulk_edit.set_storage_path( | ||||
|             [self.doc1.id], | ||||
|             self.sp1.id, | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(Document.objects.filter(storage_path=None).count(), 4) | ||||
| 
 | ||||
|         bulk_edit.set_storage_path( | ||||
|             [self.doc1.id], | ||||
|             None, | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(Document.objects.filter(storage_path=None).count(), 5) | ||||
| 
 | ||||
|         self.async_task.assert_called() | ||||
|         args, kwargs = self.async_task.call_args | ||||
| 
 | ||||
|         self.assertCountEqual(kwargs["document_ids"], [self.doc1.id]) | ||||
| 
 | ||||
|     def test_add_tag(self): | ||||
|         self.assertEqual(Document.objects.filter(tags__id=self.t1.id).count(), 2) | ||||
|         bulk_edit.add_tag( | ||||
|             [self.doc1.id, self.doc2.id, self.doc3.id, self.doc4.id], | ||||
|             self.t1.id, | ||||
|         ) | ||||
|         self.assertEqual(Document.objects.filter(tags__id=self.t1.id).count(), 4) | ||||
|         self.async_task.assert_called_once() | ||||
|         args, kwargs = self.async_task.call_args | ||||
|         self.assertCountEqual(kwargs["document_ids"], [self.doc1.id, self.doc3.id]) | ||||
| 
 | ||||
|     def test_remove_tag(self): | ||||
|         self.assertEqual(Document.objects.filter(tags__id=self.t1.id).count(), 2) | ||||
|         bulk_edit.remove_tag([self.doc1.id, self.doc3.id, self.doc4.id], self.t1.id) | ||||
|         self.assertEqual(Document.objects.filter(tags__id=self.t1.id).count(), 1) | ||||
|         self.async_task.assert_called_once() | ||||
|         args, kwargs = self.async_task.call_args | ||||
|         self.assertCountEqual(kwargs["document_ids"], [self.doc4.id]) | ||||
| 
 | ||||
|     def test_modify_tags(self): | ||||
|         tag_unrelated = Tag.objects.create(name="unrelated") | ||||
|         self.doc2.tags.add(tag_unrelated) | ||||
|         self.doc3.tags.add(tag_unrelated) | ||||
|         bulk_edit.modify_tags( | ||||
|             [self.doc2.id, self.doc3.id], | ||||
|             add_tags=[self.t2.id], | ||||
|             remove_tags=[self.t1.id], | ||||
|         ) | ||||
| 
 | ||||
|         self.assertCountEqual(list(self.doc2.tags.all()), [self.t2, tag_unrelated]) | ||||
|         self.assertCountEqual(list(self.doc3.tags.all()), [self.t2, tag_unrelated]) | ||||
| 
 | ||||
|         self.async_task.assert_called_once() | ||||
|         args, kwargs = self.async_task.call_args | ||||
|         # TODO: doc3 should not be affected, but the query for that is rather complicated | ||||
|         self.assertCountEqual(kwargs["document_ids"], [self.doc2.id, self.doc3.id]) | ||||
| 
 | ||||
|     def test_delete(self): | ||||
|         self.assertEqual(Document.objects.count(), 5) | ||||
|         bulk_edit.delete([self.doc1.id, self.doc2.id]) | ||||
|         self.assertEqual(Document.objects.count(), 3) | ||||
|         self.assertCountEqual( | ||||
|             [doc.id for doc in Document.objects.all()], | ||||
|             [self.doc3.id, self.doc4.id, self.doc5.id], | ||||
|         ) | ||||
| 
 | ||||
|     @mock.patch("documents.serialisers.bulk_edit.set_correspondent") | ||||
|     def test_api_set_correspondent(self, m): | ||||
|         m.return_value = "OK" | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc1.id], | ||||
|                     "method": "set_correspondent", | ||||
|                     "parameters": {"correspondent": self.c1.id}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         m.assert_called_once() | ||||
|         args, kwargs = m.call_args | ||||
|         self.assertEqual(args[0], [self.doc1.id]) | ||||
|         self.assertEqual(kwargs["correspondent"], self.c1.id) | ||||
| 
 | ||||
|     @mock.patch("documents.serialisers.bulk_edit.set_correspondent") | ||||
|     def test_api_unset_correspondent(self, m): | ||||
|         m.return_value = "OK" | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc1.id], | ||||
|                     "method": "set_correspondent", | ||||
|                     "parameters": {"correspondent": None}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         m.assert_called_once() | ||||
|         args, kwargs = m.call_args | ||||
|         self.assertEqual(args[0], [self.doc1.id]) | ||||
|         self.assertIsNone(kwargs["correspondent"]) | ||||
| 
 | ||||
|     @mock.patch("documents.serialisers.bulk_edit.set_document_type") | ||||
|     def test_api_set_type(self, m): | ||||
|         m.return_value = "OK" | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc1.id], | ||||
|                     "method": "set_document_type", | ||||
|                     "parameters": {"document_type": self.dt1.id}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         m.assert_called_once() | ||||
|         args, kwargs = m.call_args | ||||
|         self.assertEqual(args[0], [self.doc1.id]) | ||||
|         self.assertEqual(kwargs["document_type"], self.dt1.id) | ||||
| 
 | ||||
|     @mock.patch("documents.serialisers.bulk_edit.set_document_type") | ||||
|     def test_api_unset_type(self, m): | ||||
|         m.return_value = "OK" | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc1.id], | ||||
|                     "method": "set_document_type", | ||||
|                     "parameters": {"document_type": None}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         m.assert_called_once() | ||||
|         args, kwargs = m.call_args | ||||
|         self.assertEqual(args[0], [self.doc1.id]) | ||||
|         self.assertIsNone(kwargs["document_type"]) | ||||
| 
 | ||||
|     @mock.patch("documents.serialisers.bulk_edit.add_tag") | ||||
|     def test_api_add_tag(self, m): | ||||
|         m.return_value = "OK" | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc1.id], | ||||
|                     "method": "add_tag", | ||||
|                     "parameters": {"tag": self.t1.id}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         m.assert_called_once() | ||||
|         args, kwargs = m.call_args | ||||
|         self.assertEqual(args[0], [self.doc1.id]) | ||||
|         self.assertEqual(kwargs["tag"], self.t1.id) | ||||
| 
 | ||||
|     @mock.patch("documents.serialisers.bulk_edit.remove_tag") | ||||
|     def test_api_remove_tag(self, m): | ||||
|         m.return_value = "OK" | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc1.id], | ||||
|                     "method": "remove_tag", | ||||
|                     "parameters": {"tag": self.t1.id}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         m.assert_called_once() | ||||
|         args, kwargs = m.call_args | ||||
|         self.assertEqual(args[0], [self.doc1.id]) | ||||
|         self.assertEqual(kwargs["tag"], self.t1.id) | ||||
| 
 | ||||
|     @mock.patch("documents.serialisers.bulk_edit.modify_tags") | ||||
|     def test_api_modify_tags(self, m): | ||||
|         m.return_value = "OK" | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc1.id, self.doc3.id], | ||||
|                     "method": "modify_tags", | ||||
|                     "parameters": { | ||||
|                         "add_tags": [self.t1.id], | ||||
|                         "remove_tags": [self.t2.id], | ||||
|                     }, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         m.assert_called_once() | ||||
|         args, kwargs = m.call_args | ||||
|         self.assertListEqual(args[0], [self.doc1.id, self.doc3.id]) | ||||
|         self.assertEqual(kwargs["add_tags"], [self.t1.id]) | ||||
|         self.assertEqual(kwargs["remove_tags"], [self.t2.id]) | ||||
| 
 | ||||
|     @mock.patch("documents.serialisers.bulk_edit.modify_tags") | ||||
|     def test_api_modify_tags_not_provided(self, m): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API data to modify tags is missing modify_tags field | ||||
|         WHEN: | ||||
|             - API to edit tags is called | ||||
|         THEN: | ||||
|             - API returns HTTP 400 | ||||
|             - modify_tags is not called | ||||
|         """ | ||||
|         m.return_value = "OK" | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc1.id, self.doc3.id], | ||||
|                     "method": "modify_tags", | ||||
|                     "parameters": { | ||||
|                         "add_tags": [self.t1.id], | ||||
|                     }, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
|         m.assert_not_called() | ||||
| 
 | ||||
|     @mock.patch("documents.serialisers.bulk_edit.delete") | ||||
|     def test_api_delete(self, m): | ||||
|         m.return_value = "OK" | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 {"documents": [self.doc1.id], "method": "delete", "parameters": {}}, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         m.assert_called_once() | ||||
|         args, kwargs = m.call_args | ||||
|         self.assertEqual(args[0], [self.doc1.id]) | ||||
|         self.assertEqual(len(kwargs), 0) | ||||
| 
 | ||||
|     @mock.patch("documents.serialisers.bulk_edit.set_storage_path") | ||||
|     def test_api_set_storage_path(self, m): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API data to set the storage path of a document | ||||
|         WHEN: | ||||
|             - API is called | ||||
|         THEN: | ||||
|             - set_storage_path is called with correct document IDs and storage_path ID | ||||
|         """ | ||||
|         m.return_value = "OK" | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc1.id], | ||||
|                     "method": "set_storage_path", | ||||
|                     "parameters": {"storage_path": self.sp1.id}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         m.assert_called_once() | ||||
|         args, kwargs = m.call_args | ||||
| 
 | ||||
|         self.assertListEqual(args[0], [self.doc1.id]) | ||||
|         self.assertEqual(kwargs["storage_path"], self.sp1.id) | ||||
| 
 | ||||
|     @mock.patch("documents.serialisers.bulk_edit.set_storage_path") | ||||
|     def test_api_unset_storage_path(self, m): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API data to clear/unset the storage path of a document | ||||
|         WHEN: | ||||
|             - API is called | ||||
|         THEN: | ||||
|             - set_storage_path is called with correct document IDs and None storage_path | ||||
|         """ | ||||
|         m.return_value = "OK" | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc1.id], | ||||
|                     "method": "set_storage_path", | ||||
|                     "parameters": {"storage_path": None}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         m.assert_called_once() | ||||
|         args, kwargs = m.call_args | ||||
| 
 | ||||
|         self.assertListEqual(args[0], [self.doc1.id]) | ||||
|         self.assertEqual(kwargs["storage_path"], None) | ||||
| 
 | ||||
|     def test_api_invalid_storage_path(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API data to set the storage path of a document | ||||
|             - Given storage_path ID isn't valid | ||||
|         WHEN: | ||||
|             - API is called | ||||
|         THEN: | ||||
|             - set_storage_path is called with correct document IDs and storage_path ID | ||||
|         """ | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc1.id], | ||||
|                     "method": "set_storage_path", | ||||
|                     "parameters": {"storage_path": self.sp1.id + 10}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
|         self.async_task.assert_not_called() | ||||
| 
 | ||||
|     def test_api_set_storage_path_not_provided(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API data to set the storage path of a document | ||||
|             - API data is missing storage path ID | ||||
|         WHEN: | ||||
|             - API is called | ||||
|         THEN: | ||||
|             - set_storage_path is called with correct document IDs and storage_path ID | ||||
|         """ | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc1.id], | ||||
|                     "method": "set_storage_path", | ||||
|                     "parameters": {}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
|         self.async_task.assert_not_called() | ||||
| 
 | ||||
|     def test_api_invalid_doc(self): | ||||
|         self.assertEqual(Document.objects.count(), 5) | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps({"documents": [-235], "method": "delete", "parameters": {}}), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
|         self.assertEqual(Document.objects.count(), 5) | ||||
| 
 | ||||
|     def test_api_invalid_method(self): | ||||
|         self.assertEqual(Document.objects.count(), 5) | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc2.id], | ||||
|                     "method": "exterminate", | ||||
|                     "parameters": {}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
|         self.assertEqual(Document.objects.count(), 5) | ||||
| 
 | ||||
|     def test_api_invalid_correspondent(self): | ||||
|         self.assertEqual(self.doc2.correspondent, self.c1) | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc2.id], | ||||
|                     "method": "set_correspondent", | ||||
|                     "parameters": {"correspondent": 345657}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
| 
 | ||||
|         doc2 = Document.objects.get(id=self.doc2.id) | ||||
|         self.assertEqual(doc2.correspondent, self.c1) | ||||
| 
 | ||||
|     def test_api_no_correspondent(self): | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc2.id], | ||||
|                     "method": "set_correspondent", | ||||
|                     "parameters": {}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
| 
 | ||||
|     def test_api_invalid_document_type(self): | ||||
|         self.assertEqual(self.doc2.document_type, self.dt1) | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc2.id], | ||||
|                     "method": "set_document_type", | ||||
|                     "parameters": {"document_type": 345657}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
| 
 | ||||
|         doc2 = Document.objects.get(id=self.doc2.id) | ||||
|         self.assertEqual(doc2.document_type, self.dt1) | ||||
| 
 | ||||
|     def test_api_no_document_type(self): | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc2.id], | ||||
|                     "method": "set_document_type", | ||||
|                     "parameters": {}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
| 
 | ||||
|     def test_api_add_invalid_tag(self): | ||||
|         self.assertEqual(list(self.doc2.tags.all()), [self.t1]) | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc2.id], | ||||
|                     "method": "add_tag", | ||||
|                     "parameters": {"tag": 345657}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
| 
 | ||||
|         self.assertEqual(list(self.doc2.tags.all()), [self.t1]) | ||||
| 
 | ||||
|     def test_api_add_tag_no_tag(self): | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 {"documents": [self.doc2.id], "method": "add_tag", "parameters": {}}, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
| 
 | ||||
|     def test_api_delete_invalid_tag(self): | ||||
|         self.assertEqual(list(self.doc2.tags.all()), [self.t1]) | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc2.id], | ||||
|                     "method": "remove_tag", | ||||
|                     "parameters": {"tag": 345657}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
| 
 | ||||
|         self.assertEqual(list(self.doc2.tags.all()), [self.t1]) | ||||
| 
 | ||||
|     def test_api_delete_tag_no_tag(self): | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 {"documents": [self.doc2.id], "method": "remove_tag", "parameters": {}}, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
| 
 | ||||
|     def test_api_modify_invalid_tags(self): | ||||
|         self.assertEqual(list(self.doc2.tags.all()), [self.t1]) | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc2.id], | ||||
|                     "method": "modify_tags", | ||||
|                     "parameters": { | ||||
|                         "add_tags": [self.t2.id, 1657], | ||||
|                         "remove_tags": [1123123], | ||||
|                     }, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
| 
 | ||||
|     def test_api_modify_tags_no_tags(self): | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc2.id], | ||||
|                     "method": "modify_tags", | ||||
|                     "parameters": {"remove_tags": [1123123]}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc2.id], | ||||
|                     "method": "modify_tags", | ||||
|                     "parameters": {"add_tags": [self.t2.id, 1657]}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
| 
 | ||||
|     def test_api_selection_data_empty(self): | ||||
|         response = self.client.post( | ||||
|             "/api/documents/selection_data/", | ||||
|             json.dumps({"documents": []}), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         for field, Entity in [ | ||||
|             ("selected_correspondents", Correspondent), | ||||
|             ("selected_tags", Tag), | ||||
|             ("selected_document_types", DocumentType), | ||||
|         ]: | ||||
|             self.assertEqual(len(response.data[field]), Entity.objects.count()) | ||||
|             for correspondent in response.data[field]: | ||||
|                 self.assertEqual(correspondent["document_count"], 0) | ||||
|             self.assertCountEqual( | ||||
|                 map(lambda c: c["id"], response.data[field]), | ||||
|                 map(lambda c: c["id"], Entity.objects.values("id")), | ||||
|             ) | ||||
| 
 | ||||
|     def test_api_selection_data(self): | ||||
|         response = self.client.post( | ||||
|             "/api/documents/selection_data/", | ||||
|             json.dumps( | ||||
|                 {"documents": [self.doc1.id, self.doc2.id, self.doc4.id, self.doc5.id]}, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
| 
 | ||||
|         self.assertCountEqual( | ||||
|             response.data["selected_correspondents"], | ||||
|             [ | ||||
|                 {"id": self.c1.id, "document_count": 1}, | ||||
|                 {"id": self.c2.id, "document_count": 0}, | ||||
|             ], | ||||
|         ) | ||||
|         self.assertCountEqual( | ||||
|             response.data["selected_tags"], | ||||
|             [ | ||||
|                 {"id": self.t1.id, "document_count": 2}, | ||||
|                 {"id": self.t2.id, "document_count": 1}, | ||||
|             ], | ||||
|         ) | ||||
|         self.assertCountEqual( | ||||
|             response.data["selected_document_types"], | ||||
|             [ | ||||
|                 {"id": self.c1.id, "document_count": 1}, | ||||
|                 {"id": self.c2.id, "document_count": 0}, | ||||
|             ], | ||||
|         ) | ||||
| 
 | ||||
|     @mock.patch("documents.serialisers.bulk_edit.set_permissions") | ||||
|     def test_set_permissions(self, m): | ||||
|         m.return_value = "OK" | ||||
|         user1 = User.objects.create(username="user1") | ||||
|         user2 = User.objects.create(username="user2") | ||||
|         permissions = { | ||||
|             "view": { | ||||
|                 "users": [user1.id, user2.id], | ||||
|                 "groups": None, | ||||
|             }, | ||||
|             "change": { | ||||
|                 "users": [user1.id], | ||||
|                 "groups": None, | ||||
|             }, | ||||
|         } | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc2.id, self.doc3.id], | ||||
|                     "method": "set_permissions", | ||||
|                     "parameters": {"set_permissions": permissions}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
| 
 | ||||
|         m.assert_called_once() | ||||
|         args, kwargs = m.call_args | ||||
|         self.assertCountEqual(args[0], [self.doc2.id, self.doc3.id]) | ||||
|         self.assertEqual(len(kwargs["set_permissions"]["view"]["users"]), 2) | ||||
| 
 | ||||
|     @mock.patch("documents.serialisers.bulk_edit.set_permissions") | ||||
|     def test_insufficient_permissions_ownership(self, m): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Documents owned by user other than logged in user | ||||
|         WHEN: | ||||
|             - set_permissions bulk edit API endpoint is called | ||||
|         THEN: | ||||
|             - User is not able to change permissions | ||||
|         """ | ||||
|         m.return_value = "OK" | ||||
|         self.doc1.owner = User.objects.get(username="temp_admin") | ||||
|         self.doc1.save() | ||||
|         user1 = User.objects.create(username="user1") | ||||
|         self.client.force_authenticate(user=user1) | ||||
| 
 | ||||
|         permissions = { | ||||
|             "owner": user1.id, | ||||
|         } | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc1.id, self.doc2.id, self.doc3.id], | ||||
|                     "method": "set_permissions", | ||||
|                     "parameters": {"set_permissions": permissions}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_403_FORBIDDEN) | ||||
| 
 | ||||
|         m.assert_not_called() | ||||
|         self.assertEqual(response.content, b"Insufficient permissions") | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc2.id, self.doc3.id], | ||||
|                     "method": "set_permissions", | ||||
|                     "parameters": {"set_permissions": permissions}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         m.assert_called_once() | ||||
| 
 | ||||
|     @mock.patch("documents.serialisers.bulk_edit.set_storage_path") | ||||
|     def test_insufficient_permissions_edit(self, m): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Documents for which current user only has view permissions | ||||
|         WHEN: | ||||
|             - API is called | ||||
|         THEN: | ||||
|             - set_storage_path only called if user can edit all docs | ||||
|         """ | ||||
|         m.return_value = "OK" | ||||
|         self.doc1.owner = User.objects.get(username="temp_admin") | ||||
|         self.doc1.save() | ||||
|         user1 = User.objects.create(username="user1") | ||||
|         assign_perm("view_document", user1, self.doc1) | ||||
|         self.client.force_authenticate(user=user1) | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc1.id, self.doc2.id, self.doc3.id], | ||||
|                     "method": "set_storage_path", | ||||
|                     "parameters": {"storage_path": self.sp1.id}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_403_FORBIDDEN) | ||||
| 
 | ||||
|         m.assert_not_called() | ||||
|         self.assertEqual(response.content, b"Insufficient permissions") | ||||
| 
 | ||||
|         assign_perm("change_document", user1, self.doc1) | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/documents/bulk_edit/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "documents": [self.doc1.id, self.doc2.id, self.doc3.id], | ||||
|                     "method": "set_storage_path", | ||||
|                     "parameters": {"storage_path": self.sp1.id}, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
| 
 | ||||
|         m.assert_called_once() | ||||
							
								
								
									
										236
									
								
								src/documents/tests/test_api_consumption_templates.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										236
									
								
								src/documents/tests/test_api_consumption_templates.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,236 @@ | ||||
| import json | ||||
| 
 | ||||
| from django.contrib.auth.models import Group | ||||
| from django.contrib.auth.models import User | ||||
| from rest_framework import status | ||||
| from rest_framework.test import APITestCase | ||||
| 
 | ||||
| from documents.data_models import DocumentSource | ||||
| from documents.models import ConsumptionTemplate | ||||
| from documents.models import Correspondent | ||||
| from documents.models import CustomField | ||||
| from documents.models import DocumentType | ||||
| from documents.models import StoragePath | ||||
| from documents.models import Tag | ||||
| from documents.tests.utils import DirectoriesMixin | ||||
| from paperless_mail.models import MailAccount | ||||
| from paperless_mail.models import MailRule | ||||
| 
 | ||||
| 
 | ||||
| class TestApiConsumptionTemplates(DirectoriesMixin, APITestCase): | ||||
|     ENDPOINT = "/api/consumption_templates/" | ||||
| 
 | ||||
|     def setUp(self) -> None: | ||||
|         super().setUp() | ||||
| 
 | ||||
|         user = User.objects.create_superuser(username="temp_admin") | ||||
|         self.client.force_authenticate(user=user) | ||||
|         self.user2 = User.objects.create(username="user2") | ||||
|         self.user3 = User.objects.create(username="user3") | ||||
|         self.group1 = Group.objects.create(name="group1") | ||||
| 
 | ||||
|         self.c = Correspondent.objects.create(name="Correspondent Name") | ||||
|         self.c2 = Correspondent.objects.create(name="Correspondent Name 2") | ||||
|         self.dt = DocumentType.objects.create(name="DocType Name") | ||||
|         self.t1 = Tag.objects.create(name="t1") | ||||
|         self.t2 = Tag.objects.create(name="t2") | ||||
|         self.t3 = Tag.objects.create(name="t3") | ||||
|         self.sp = StoragePath.objects.create(path="/test/") | ||||
|         self.cf1 = CustomField.objects.create(name="Custom Field 1", data_type="string") | ||||
|         self.cf2 = CustomField.objects.create( | ||||
|             name="Custom Field 2", | ||||
|             data_type="integer", | ||||
|         ) | ||||
| 
 | ||||
|         self.ct = ConsumptionTemplate.objects.create( | ||||
|             name="Template 1", | ||||
|             order=0, | ||||
|             sources=f"{int(DocumentSource.ApiUpload)},{int(DocumentSource.ConsumeFolder)},{int(DocumentSource.MailFetch)}", | ||||
|             filter_filename="*simple*", | ||||
|             filter_path="*/samples/*", | ||||
|             assign_title="Doc from {correspondent}", | ||||
|             assign_correspondent=self.c, | ||||
|             assign_document_type=self.dt, | ||||
|             assign_storage_path=self.sp, | ||||
|             assign_owner=self.user2, | ||||
|         ) | ||||
|         self.ct.assign_tags.add(self.t1) | ||||
|         self.ct.assign_tags.add(self.t2) | ||||
|         self.ct.assign_tags.add(self.t3) | ||||
|         self.ct.assign_view_users.add(self.user3.pk) | ||||
|         self.ct.assign_view_groups.add(self.group1.pk) | ||||
|         self.ct.assign_change_users.add(self.user3.pk) | ||||
|         self.ct.assign_change_groups.add(self.group1.pk) | ||||
|         self.ct.assign_custom_fields.add(self.cf1.pk) | ||||
|         self.ct.assign_custom_fields.add(self.cf2.pk) | ||||
|         self.ct.save() | ||||
| 
 | ||||
|     def test_api_get_consumption_template(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API request to get all consumption template | ||||
|         WHEN: | ||||
|             - API is called | ||||
|         THEN: | ||||
|             - Existing consumption templates are returned | ||||
|         """ | ||||
|         response = self.client.get(self.ENDPOINT, format="json") | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(response.data["count"], 1) | ||||
| 
 | ||||
|         resp_consumption_template = response.data["results"][0] | ||||
|         self.assertEqual(resp_consumption_template["id"], self.ct.id) | ||||
|         self.assertEqual( | ||||
|             resp_consumption_template["assign_correspondent"], | ||||
|             self.ct.assign_correspondent.pk, | ||||
|         ) | ||||
| 
 | ||||
|     def test_api_create_consumption_template(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API request to create a consumption template | ||||
|         WHEN: | ||||
|             - API is called | ||||
|         THEN: | ||||
|             - Correct HTTP response | ||||
|             - New template is created | ||||
|         """ | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "name": "Template 2", | ||||
|                     "order": 1, | ||||
|                     "sources": [DocumentSource.ApiUpload], | ||||
|                     "filter_filename": "*test*", | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_201_CREATED) | ||||
|         self.assertEqual(ConsumptionTemplate.objects.count(), 2) | ||||
| 
 | ||||
|     def test_api_create_invalid_consumption_template(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API request to create a consumption template | ||||
|             - Neither file name nor path filter are specified | ||||
|         WHEN: | ||||
|             - API is called | ||||
|         THEN: | ||||
|             - Correct HTTP 400 response | ||||
|             - No template is created | ||||
|         """ | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "name": "Template 2", | ||||
|                     "order": 1, | ||||
|                     "sources": [DocumentSource.ApiUpload], | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
|         self.assertEqual(ConsumptionTemplate.objects.count(), 1) | ||||
| 
 | ||||
|     def test_api_create_consumption_template_empty_fields(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API request to create a consumption template | ||||
|             - Path or filename filter or assign title are empty string | ||||
|         WHEN: | ||||
|             - API is called | ||||
|         THEN: | ||||
|             - Template is created but filter or title assignment is not set if "" | ||||
|         """ | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "name": "Template 2", | ||||
|                     "order": 1, | ||||
|                     "sources": [DocumentSource.ApiUpload], | ||||
|                     "filter_filename": "*test*", | ||||
|                     "filter_path": "", | ||||
|                     "assign_title": "", | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_201_CREATED) | ||||
|         ct = ConsumptionTemplate.objects.get(name="Template 2") | ||||
|         self.assertEqual(ct.filter_filename, "*test*") | ||||
|         self.assertIsNone(ct.filter_path) | ||||
|         self.assertIsNone(ct.assign_title) | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "name": "Template 3", | ||||
|                     "order": 1, | ||||
|                     "sources": [DocumentSource.ApiUpload], | ||||
|                     "filter_filename": "", | ||||
|                     "filter_path": "*/test/*", | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_201_CREATED) | ||||
|         ct2 = ConsumptionTemplate.objects.get(name="Template 3") | ||||
|         self.assertEqual(ct2.filter_path, "*/test/*") | ||||
|         self.assertIsNone(ct2.filter_filename) | ||||
| 
 | ||||
|     def test_api_create_consumption_template_with_mailrule(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API request to create a consumption template with a mail rule but no MailFetch source | ||||
|         WHEN: | ||||
|             - API is called | ||||
|         THEN: | ||||
|             - New template is created with MailFetch as source | ||||
|         """ | ||||
|         account1 = MailAccount.objects.create( | ||||
|             name="Email1", | ||||
|             username="username1", | ||||
|             password="password1", | ||||
|             imap_server="server.example.com", | ||||
|             imap_port=443, | ||||
|             imap_security=MailAccount.ImapSecurity.SSL, | ||||
|             character_set="UTF-8", | ||||
|         ) | ||||
|         rule1 = MailRule.objects.create( | ||||
|             name="Rule1", | ||||
|             account=account1, | ||||
|             folder="INBOX", | ||||
|             filter_from="from@example.com", | ||||
|             filter_to="someone@somewhere.com", | ||||
|             filter_subject="subject", | ||||
|             filter_body="body", | ||||
|             filter_attachment_filename_include="file.pdf", | ||||
|             maximum_age=30, | ||||
|             action=MailRule.MailAction.MARK_READ, | ||||
|             assign_title_from=MailRule.TitleSource.FROM_SUBJECT, | ||||
|             assign_correspondent_from=MailRule.CorrespondentSource.FROM_NOTHING, | ||||
|             order=0, | ||||
|             attachment_type=MailRule.AttachmentProcessing.ATTACHMENTS_ONLY, | ||||
|         ) | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "name": "Template 2", | ||||
|                     "order": 1, | ||||
|                     "sources": [DocumentSource.ApiUpload], | ||||
|                     "filter_mailrule": rule1.pk, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_201_CREATED) | ||||
|         self.assertEqual(ConsumptionTemplate.objects.count(), 2) | ||||
|         ct = ConsumptionTemplate.objects.get(name="Template 2") | ||||
|         self.assertEqual(ct.sources, [int(DocumentSource.MailFetch).__str__()]) | ||||
							
								
								
									
										1992
									
								
								src/documents/tests/test_api_documents.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1992
									
								
								src/documents/tests/test_api_documents.py
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										224
									
								
								src/documents/tests/test_api_objects.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										224
									
								
								src/documents/tests/test_api_objects.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,224 @@ | ||||
| import json | ||||
| from unittest import mock | ||||
| 
 | ||||
| from django.contrib.auth.models import User | ||||
| from rest_framework import status | ||||
| from rest_framework.test import APITestCase | ||||
| 
 | ||||
| from documents.models import Correspondent | ||||
| from documents.models import Document | ||||
| from documents.models import DocumentType | ||||
| from documents.models import StoragePath | ||||
| from documents.models import Tag | ||||
| from documents.tests.utils import DirectoriesMixin | ||||
| 
 | ||||
| 
 | ||||
| class TestApiObjects(DirectoriesMixin, APITestCase): | ||||
|     def setUp(self) -> None: | ||||
|         super().setUp() | ||||
| 
 | ||||
|         user = User.objects.create_superuser(username="temp_admin") | ||||
|         self.client.force_authenticate(user=user) | ||||
| 
 | ||||
|         self.tag1 = Tag.objects.create(name="t1", is_inbox_tag=True) | ||||
|         self.tag2 = Tag.objects.create(name="t2") | ||||
|         self.tag3 = Tag.objects.create(name="t3") | ||||
|         self.c1 = Correspondent.objects.create(name="c1") | ||||
|         self.c2 = Correspondent.objects.create(name="c2") | ||||
|         self.c3 = Correspondent.objects.create(name="c3") | ||||
|         self.dt1 = DocumentType.objects.create(name="dt1") | ||||
|         self.dt2 = DocumentType.objects.create(name="dt2") | ||||
|         self.sp1 = StoragePath.objects.create(name="sp1", path="Something/{title}") | ||||
|         self.sp2 = StoragePath.objects.create(name="sp2", path="Something2/{title}") | ||||
| 
 | ||||
|     def test_object_filters(self): | ||||
|         response = self.client.get( | ||||
|             f"/api/tags/?id={self.tag2.id}", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         results = response.data["results"] | ||||
|         self.assertEqual(len(results), 1) | ||||
| 
 | ||||
|         response = self.client.get( | ||||
|             f"/api/tags/?id__in={self.tag1.id},{self.tag3.id}", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         results = response.data["results"] | ||||
|         self.assertEqual(len(results), 2) | ||||
| 
 | ||||
|         response = self.client.get( | ||||
|             f"/api/correspondents/?id={self.c2.id}", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         results = response.data["results"] | ||||
|         self.assertEqual(len(results), 1) | ||||
| 
 | ||||
|         response = self.client.get( | ||||
|             f"/api/correspondents/?id__in={self.c1.id},{self.c3.id}", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         results = response.data["results"] | ||||
|         self.assertEqual(len(results), 2) | ||||
| 
 | ||||
|         response = self.client.get( | ||||
|             f"/api/document_types/?id={self.dt1.id}", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         results = response.data["results"] | ||||
|         self.assertEqual(len(results), 1) | ||||
| 
 | ||||
|         response = self.client.get( | ||||
|             f"/api/document_types/?id__in={self.dt1.id},{self.dt2.id}", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         results = response.data["results"] | ||||
|         self.assertEqual(len(results), 2) | ||||
| 
 | ||||
|         response = self.client.get( | ||||
|             f"/api/storage_paths/?id={self.sp1.id}", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         results = response.data["results"] | ||||
|         self.assertEqual(len(results), 1) | ||||
| 
 | ||||
|         response = self.client.get( | ||||
|             f"/api/storage_paths/?id__in={self.sp1.id},{self.sp2.id}", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         results = response.data["results"] | ||||
|         self.assertEqual(len(results), 2) | ||||
| 
 | ||||
| 
 | ||||
| class TestApiStoragePaths(DirectoriesMixin, APITestCase): | ||||
|     ENDPOINT = "/api/storage_paths/" | ||||
| 
 | ||||
|     def setUp(self) -> None: | ||||
|         super().setUp() | ||||
| 
 | ||||
|         user = User.objects.create_superuser(username="temp_admin") | ||||
|         self.client.force_authenticate(user=user) | ||||
| 
 | ||||
|         self.sp1 = StoragePath.objects.create(name="sp1", path="Something/{checksum}") | ||||
| 
 | ||||
|     def test_api_get_storage_path(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API request to get all storage paths | ||||
|         WHEN: | ||||
|             - API is called | ||||
|         THEN: | ||||
|             - Existing storage paths are returned | ||||
|         """ | ||||
|         response = self.client.get(self.ENDPOINT, format="json") | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(response.data["count"], 1) | ||||
| 
 | ||||
|         resp_storage_path = response.data["results"][0] | ||||
|         self.assertEqual(resp_storage_path["id"], self.sp1.id) | ||||
|         self.assertEqual(resp_storage_path["path"], self.sp1.path) | ||||
| 
 | ||||
|     def test_api_create_storage_path(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API request to create a storage paths | ||||
|         WHEN: | ||||
|             - API is called | ||||
|         THEN: | ||||
|             - Correct HTTP response | ||||
|             - New storage path is created | ||||
|         """ | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "name": "A storage path", | ||||
|                     "path": "Somewhere/{asn}", | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_201_CREATED) | ||||
|         self.assertEqual(StoragePath.objects.count(), 2) | ||||
| 
 | ||||
|     def test_api_create_invalid_storage_path(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API request to create a storage paths | ||||
|             - Storage path format is incorrect | ||||
|         WHEN: | ||||
|             - API is called | ||||
|         THEN: | ||||
|             - Correct HTTP 400 response | ||||
|             - No storage path is created | ||||
|         """ | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "name": "Another storage path", | ||||
|                     "path": "Somewhere/{correspdent}", | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
|         self.assertEqual(StoragePath.objects.count(), 1) | ||||
| 
 | ||||
|     def test_api_storage_path_placeholders(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API request to create a storage path with placeholders | ||||
|             - Storage path is valid | ||||
|         WHEN: | ||||
|             - API is called | ||||
|         THEN: | ||||
|             - Correct HTTP response | ||||
|             - New storage path is created | ||||
|         """ | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "name": "Storage path with placeholders", | ||||
|                     "path": "{title}/{correspondent}/{document_type}/{created}/{created_year}" | ||||
|                     "/{created_year_short}/{created_month}/{created_month_name}" | ||||
|                     "/{created_month_name_short}/{created_day}/{added}/{added_year}" | ||||
|                     "/{added_year_short}/{added_month}/{added_month_name}" | ||||
|                     "/{added_month_name_short}/{added_day}/{asn}/{tags}" | ||||
|                     "/{tag_list}/{owner_username}/{original_name}/{doc_pk}/", | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_201_CREATED) | ||||
|         self.assertEqual(StoragePath.objects.count(), 2) | ||||
| 
 | ||||
|     @mock.patch("documents.bulk_edit.bulk_update_documents.delay") | ||||
|     def test_api_update_storage_path(self, bulk_update_mock): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API request to get all storage paths | ||||
|         WHEN: | ||||
|             - API is called | ||||
|         THEN: | ||||
|             - Existing storage paths are returned | ||||
|         """ | ||||
|         document = Document.objects.create( | ||||
|             mime_type="application/pdf", | ||||
|             storage_path=self.sp1, | ||||
|         ) | ||||
|         response = self.client.patch( | ||||
|             f"{self.ENDPOINT}{self.sp1.pk}/", | ||||
|             data={ | ||||
|                 "path": "somewhere/{created} - {title}", | ||||
|             }, | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
| 
 | ||||
|         bulk_update_mock.assert_called_once() | ||||
| 
 | ||||
|         args, _ = bulk_update_mock.call_args | ||||
| 
 | ||||
|         self.assertCountEqual([document.pk], args[0]) | ||||
							
								
								
									
										910
									
								
								src/documents/tests/test_api_permissions.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										910
									
								
								src/documents/tests/test_api_permissions.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,910 @@ | ||||
| import json | ||||
| 
 | ||||
| from django.contrib.auth.models import Group | ||||
| from django.contrib.auth.models import Permission | ||||
| from django.contrib.auth.models import User | ||||
| from guardian.shortcuts import assign_perm | ||||
| from guardian.shortcuts import get_perms | ||||
| from guardian.shortcuts import get_users_with_perms | ||||
| from rest_framework import status | ||||
| from rest_framework.test import APITestCase | ||||
| 
 | ||||
| from documents.models import Correspondent | ||||
| from documents.models import Document | ||||
| from documents.models import DocumentType | ||||
| from documents.models import MatchingModel | ||||
| from documents.models import StoragePath | ||||
| from documents.models import Tag | ||||
| from documents.tests.utils import DirectoriesMixin | ||||
| 
 | ||||
| 
 | ||||
| class TestApiAuth(DirectoriesMixin, APITestCase): | ||||
|     def test_auth_required(self): | ||||
|         d = Document.objects.create(title="Test") | ||||
| 
 | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/documents/").status_code, | ||||
|             status.HTTP_401_UNAUTHORIZED, | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual( | ||||
|             self.client.get(f"/api/documents/{d.id}/").status_code, | ||||
|             status.HTTP_401_UNAUTHORIZED, | ||||
|         ) | ||||
|         self.assertEqual( | ||||
|             self.client.get(f"/api/documents/{d.id}/download/").status_code, | ||||
|             status.HTTP_401_UNAUTHORIZED, | ||||
|         ) | ||||
|         self.assertEqual( | ||||
|             self.client.get(f"/api/documents/{d.id}/preview/").status_code, | ||||
|             status.HTTP_401_UNAUTHORIZED, | ||||
|         ) | ||||
|         self.assertEqual( | ||||
|             self.client.get(f"/api/documents/{d.id}/thumb/").status_code, | ||||
|             status.HTTP_401_UNAUTHORIZED, | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/tags/").status_code, | ||||
|             status.HTTP_401_UNAUTHORIZED, | ||||
|         ) | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/correspondents/").status_code, | ||||
|             status.HTTP_401_UNAUTHORIZED, | ||||
|         ) | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/document_types/").status_code, | ||||
|             status.HTTP_401_UNAUTHORIZED, | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/logs/").status_code, | ||||
|             status.HTTP_401_UNAUTHORIZED, | ||||
|         ) | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/saved_views/").status_code, | ||||
|             status.HTTP_401_UNAUTHORIZED, | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/search/autocomplete/").status_code, | ||||
|             status.HTTP_401_UNAUTHORIZED, | ||||
|         ) | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/documents/bulk_edit/").status_code, | ||||
|             status.HTTP_401_UNAUTHORIZED, | ||||
|         ) | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/documents/bulk_download/").status_code, | ||||
|             status.HTTP_401_UNAUTHORIZED, | ||||
|         ) | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/documents/selection_data/").status_code, | ||||
|             status.HTTP_401_UNAUTHORIZED, | ||||
|         ) | ||||
| 
 | ||||
|     def test_api_version_no_auth(self): | ||||
|         response = self.client.get("/api/") | ||||
|         self.assertNotIn("X-Api-Version", response) | ||||
|         self.assertNotIn("X-Version", response) | ||||
| 
 | ||||
|     def test_api_version_with_auth(self): | ||||
|         user = User.objects.create_superuser(username="test") | ||||
|         self.client.force_authenticate(user) | ||||
|         response = self.client.get("/api/") | ||||
|         self.assertIn("X-Api-Version", response) | ||||
|         self.assertIn("X-Version", response) | ||||
| 
 | ||||
|     def test_api_insufficient_permissions(self): | ||||
|         user = User.objects.create_user(username="test") | ||||
|         self.client.force_authenticate(user) | ||||
| 
 | ||||
|         Document.objects.create(title="Test") | ||||
| 
 | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/documents/").status_code, | ||||
|             status.HTTP_403_FORBIDDEN, | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/tags/").status_code, | ||||
|             status.HTTP_403_FORBIDDEN, | ||||
|         ) | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/correspondents/").status_code, | ||||
|             status.HTTP_403_FORBIDDEN, | ||||
|         ) | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/document_types/").status_code, | ||||
|             status.HTTP_403_FORBIDDEN, | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/logs/").status_code, | ||||
|             status.HTTP_403_FORBIDDEN, | ||||
|         ) | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/saved_views/").status_code, | ||||
|             status.HTTP_403_FORBIDDEN, | ||||
|         ) | ||||
| 
 | ||||
|     def test_api_sufficient_permissions(self): | ||||
|         user = User.objects.create_user(username="test") | ||||
|         user.user_permissions.add(*Permission.objects.all()) | ||||
|         self.client.force_authenticate(user) | ||||
| 
 | ||||
|         Document.objects.create(title="Test") | ||||
| 
 | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/documents/").status_code, | ||||
|             status.HTTP_200_OK, | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(self.client.get("/api/tags/").status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/correspondents/").status_code, | ||||
|             status.HTTP_200_OK, | ||||
|         ) | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/document_types/").status_code, | ||||
|             status.HTTP_200_OK, | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(self.client.get("/api/logs/").status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/saved_views/").status_code, | ||||
|             status.HTTP_200_OK, | ||||
|         ) | ||||
| 
 | ||||
|     def test_api_get_object_permissions(self): | ||||
|         user1 = User.objects.create_user(username="test1") | ||||
|         user2 = User.objects.create_user(username="test2") | ||||
|         user1.user_permissions.add(*Permission.objects.filter(codename="view_document")) | ||||
|         self.client.force_authenticate(user1) | ||||
| 
 | ||||
|         self.assertEqual( | ||||
|             self.client.get("/api/documents/").status_code, | ||||
|             status.HTTP_200_OK, | ||||
|         ) | ||||
| 
 | ||||
|         d = Document.objects.create(title="Test", content="the content 1", checksum="1") | ||||
| 
 | ||||
|         # no owner | ||||
|         self.assertEqual( | ||||
|             self.client.get(f"/api/documents/{d.id}/").status_code, | ||||
|             status.HTTP_200_OK, | ||||
|         ) | ||||
| 
 | ||||
|         d2 = Document.objects.create( | ||||
|             title="Test 2", | ||||
|             content="the content 2", | ||||
|             checksum="2", | ||||
|             owner=user2, | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual( | ||||
|             self.client.get(f"/api/documents/{d2.id}/").status_code, | ||||
|             status.HTTP_404_NOT_FOUND, | ||||
|         ) | ||||
| 
 | ||||
|     def test_api_default_owner(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API request to create an object (Tag) | ||||
|         WHEN: | ||||
|             - owner is not set at all | ||||
|         THEN: | ||||
|             - Object created with current user as owner | ||||
|         """ | ||||
|         user1 = User.objects.create_superuser(username="user1") | ||||
| 
 | ||||
|         self.client.force_authenticate(user1) | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/tags/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "name": "test1", | ||||
|                     "matching_algorithm": MatchingModel.MATCH_AUTO, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_201_CREATED) | ||||
| 
 | ||||
|         tag1 = Tag.objects.filter(name="test1").first() | ||||
|         self.assertEqual(tag1.owner, user1) | ||||
| 
 | ||||
|     def test_api_set_no_owner(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API request to create an object (Tag) | ||||
|         WHEN: | ||||
|             - owner is passed as None | ||||
|         THEN: | ||||
|             - Object created with no owner | ||||
|         """ | ||||
|         user1 = User.objects.create_superuser(username="user1") | ||||
| 
 | ||||
|         self.client.force_authenticate(user1) | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/tags/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "name": "test1", | ||||
|                     "matching_algorithm": MatchingModel.MATCH_AUTO, | ||||
|                     "owner": None, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_201_CREATED) | ||||
| 
 | ||||
|         tag1 = Tag.objects.filter(name="test1").first() | ||||
|         self.assertEqual(tag1.owner, None) | ||||
| 
 | ||||
|     def test_api_set_owner_w_permissions(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API request to create an object (Tag) that supplies set_permissions object | ||||
|         WHEN: | ||||
|             - owner is passed as user id | ||||
|             - view > users is set & view > groups is set | ||||
|         THEN: | ||||
|             - Object permissions are set appropriately | ||||
|         """ | ||||
|         user1 = User.objects.create_superuser(username="user1") | ||||
|         user2 = User.objects.create(username="user2") | ||||
|         group1 = Group.objects.create(name="group1") | ||||
| 
 | ||||
|         self.client.force_authenticate(user1) | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/tags/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "name": "test1", | ||||
|                     "matching_algorithm": MatchingModel.MATCH_AUTO, | ||||
|                     "owner": user1.id, | ||||
|                     "set_permissions": { | ||||
|                         "view": { | ||||
|                             "users": [user2.id], | ||||
|                             "groups": [group1.id], | ||||
|                         }, | ||||
|                         "change": { | ||||
|                             "users": None, | ||||
|                             "groups": None, | ||||
|                         }, | ||||
|                     }, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_201_CREATED) | ||||
| 
 | ||||
|         tag1 = Tag.objects.filter(name="test1").first() | ||||
| 
 | ||||
|         from guardian.core import ObjectPermissionChecker | ||||
| 
 | ||||
|         checker = ObjectPermissionChecker(user2) | ||||
|         self.assertEqual(checker.has_perm("view_tag", tag1), True) | ||||
|         self.assertIn("view_tag", get_perms(group1, tag1)) | ||||
| 
 | ||||
|     def test_api_set_other_owner_w_permissions(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API request to create an object (Tag) | ||||
|         WHEN: | ||||
|             - a different owner than is logged in is set | ||||
|             - view > groups is set | ||||
|         THEN: | ||||
|             - Object permissions are set appropriately | ||||
|         """ | ||||
|         user1 = User.objects.create_superuser(username="user1") | ||||
|         user2 = User.objects.create(username="user2") | ||||
|         group1 = Group.objects.create(name="group1") | ||||
| 
 | ||||
|         self.client.force_authenticate(user1) | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/tags/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "name": "test1", | ||||
|                     "matching_algorithm": MatchingModel.MATCH_AUTO, | ||||
|                     "owner": user2.id, | ||||
|                     "set_permissions": { | ||||
|                         "view": { | ||||
|                             "users": None, | ||||
|                             "groups": [group1.id], | ||||
|                         }, | ||||
|                         "change": { | ||||
|                             "users": None, | ||||
|                             "groups": None, | ||||
|                         }, | ||||
|                     }, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_201_CREATED) | ||||
| 
 | ||||
|         tag1 = Tag.objects.filter(name="test1").first() | ||||
| 
 | ||||
|         self.assertEqual(tag1.owner, user2) | ||||
|         self.assertIn("view_tag", get_perms(group1, tag1)) | ||||
| 
 | ||||
|     def test_api_set_doc_permissions(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - API request to update doc permissions and owner | ||||
|         WHEN: | ||||
|             - owner is set | ||||
|             - view > users is set & view > groups is set | ||||
|         THEN: | ||||
|             - Object permissions are set appropriately | ||||
|         """ | ||||
|         doc = Document.objects.create( | ||||
|             title="test", | ||||
|             mime_type="application/pdf", | ||||
|             content="this is a document", | ||||
|         ) | ||||
|         user1 = User.objects.create_superuser(username="user1") | ||||
|         user2 = User.objects.create(username="user2") | ||||
|         group1 = Group.objects.create(name="group1") | ||||
| 
 | ||||
|         self.client.force_authenticate(user1) | ||||
| 
 | ||||
|         response = self.client.patch( | ||||
|             f"/api/documents/{doc.id}/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "owner": user1.id, | ||||
|                     "set_permissions": { | ||||
|                         "view": { | ||||
|                             "users": [user2.id], | ||||
|                             "groups": [group1.id], | ||||
|                         }, | ||||
|                         "change": { | ||||
|                             "users": None, | ||||
|                             "groups": None, | ||||
|                         }, | ||||
|                     }, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         doc = Document.objects.get(pk=doc.id) | ||||
| 
 | ||||
|         self.assertEqual(doc.owner, user1) | ||||
|         from guardian.core import ObjectPermissionChecker | ||||
| 
 | ||||
|         checker = ObjectPermissionChecker(user2) | ||||
|         self.assertTrue(checker.has_perm("view_document", doc)) | ||||
|         self.assertIn("view_document", get_perms(group1, doc)) | ||||
| 
 | ||||
|     def test_dynamic_permissions_fields(self): | ||||
|         user1 = User.objects.create_user(username="user1") | ||||
|         user1.user_permissions.add(*Permission.objects.filter(codename="view_document")) | ||||
|         user2 = User.objects.create_user(username="user2") | ||||
| 
 | ||||
|         Document.objects.create(title="Test", content="content 1", checksum="1") | ||||
|         doc2 = Document.objects.create( | ||||
|             title="Test2", | ||||
|             content="content 2", | ||||
|             checksum="2", | ||||
|             owner=user2, | ||||
|         ) | ||||
|         doc3 = Document.objects.create( | ||||
|             title="Test3", | ||||
|             content="content 3", | ||||
|             checksum="3", | ||||
|             owner=user2, | ||||
|         ) | ||||
| 
 | ||||
|         assign_perm("view_document", user1, doc2) | ||||
|         assign_perm("view_document", user1, doc3) | ||||
|         assign_perm("change_document", user1, doc3) | ||||
| 
 | ||||
|         self.client.force_authenticate(user1) | ||||
| 
 | ||||
|         response = self.client.get( | ||||
|             "/api/documents/", | ||||
|             format="json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
| 
 | ||||
|         resp_data = response.json() | ||||
| 
 | ||||
|         self.assertNotIn("permissions", resp_data["results"][0]) | ||||
|         self.assertIn("user_can_change", resp_data["results"][0]) | ||||
|         self.assertEqual(resp_data["results"][0]["user_can_change"], True)  # doc1 | ||||
|         self.assertEqual(resp_data["results"][1]["user_can_change"], False)  # doc2 | ||||
|         self.assertEqual(resp_data["results"][2]["user_can_change"], True)  # doc3 | ||||
| 
 | ||||
|         response = self.client.get( | ||||
|             "/api/documents/?full_perms=true", | ||||
|             format="json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
| 
 | ||||
|         resp_data = response.json() | ||||
| 
 | ||||
|         self.assertIn("permissions", resp_data["results"][0]) | ||||
|         self.assertNotIn("user_can_change", resp_data["results"][0]) | ||||
| 
 | ||||
| 
 | ||||
| class TestApiUser(DirectoriesMixin, APITestCase): | ||||
|     ENDPOINT = "/api/users/" | ||||
| 
 | ||||
|     def setUp(self): | ||||
|         super().setUp() | ||||
| 
 | ||||
|         self.user = User.objects.create_superuser(username="temp_admin") | ||||
|         self.client.force_authenticate(user=self.user) | ||||
| 
 | ||||
|     def test_get_users(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Configured users | ||||
|         WHEN: | ||||
|             - API call is made to get users | ||||
|         THEN: | ||||
|             - Configured users are provided | ||||
|         """ | ||||
| 
 | ||||
|         user1 = User.objects.create( | ||||
|             username="testuser", | ||||
|             password="test", | ||||
|             first_name="Test", | ||||
|             last_name="User", | ||||
|         ) | ||||
| 
 | ||||
|         response = self.client.get(self.ENDPOINT) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(response.data["count"], 2) | ||||
|         returned_user2 = response.data["results"][1] | ||||
| 
 | ||||
|         self.assertEqual(returned_user2["username"], user1.username) | ||||
|         self.assertEqual(returned_user2["password"], "**********") | ||||
|         self.assertEqual(returned_user2["first_name"], user1.first_name) | ||||
|         self.assertEqual(returned_user2["last_name"], user1.last_name) | ||||
| 
 | ||||
|     def test_create_user(self): | ||||
|         """ | ||||
|         WHEN: | ||||
|             - API request is made to add a user account | ||||
|         THEN: | ||||
|             - A new user account is created | ||||
|         """ | ||||
| 
 | ||||
|         user1 = { | ||||
|             "username": "testuser", | ||||
|             "password": "test", | ||||
|             "first_name": "Test", | ||||
|             "last_name": "User", | ||||
|         } | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             data=user1, | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_201_CREATED) | ||||
| 
 | ||||
|         returned_user1 = User.objects.get(username="testuser") | ||||
| 
 | ||||
|         self.assertEqual(returned_user1.username, user1["username"]) | ||||
|         self.assertEqual(returned_user1.first_name, user1["first_name"]) | ||||
|         self.assertEqual(returned_user1.last_name, user1["last_name"]) | ||||
| 
 | ||||
|     def test_delete_user(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Existing user account | ||||
|         WHEN: | ||||
|             - API request is made to delete a user account | ||||
|         THEN: | ||||
|             - Account is deleted | ||||
|         """ | ||||
| 
 | ||||
|         user1 = User.objects.create( | ||||
|             username="testuser", | ||||
|             password="test", | ||||
|             first_name="Test", | ||||
|             last_name="User", | ||||
|         ) | ||||
| 
 | ||||
|         nUsers = User.objects.count() | ||||
| 
 | ||||
|         response = self.client.delete( | ||||
|             f"{self.ENDPOINT}{user1.pk}/", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_204_NO_CONTENT) | ||||
| 
 | ||||
|         self.assertEqual(User.objects.count(), nUsers - 1) | ||||
| 
 | ||||
|     def test_update_user(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Existing user accounts | ||||
|         WHEN: | ||||
|             - API request is made to update user account | ||||
|         THEN: | ||||
|             - The user account is updated, password only updated if not '****' | ||||
|         """ | ||||
| 
 | ||||
|         user1 = User.objects.create( | ||||
|             username="testuser", | ||||
|             password="test", | ||||
|             first_name="Test", | ||||
|             last_name="User", | ||||
|         ) | ||||
| 
 | ||||
|         initial_password = user1.password | ||||
| 
 | ||||
|         response = self.client.patch( | ||||
|             f"{self.ENDPOINT}{user1.pk}/", | ||||
|             data={ | ||||
|                 "first_name": "Updated Name 1", | ||||
|                 "password": "******", | ||||
|             }, | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
| 
 | ||||
|         returned_user1 = User.objects.get(pk=user1.pk) | ||||
|         self.assertEqual(returned_user1.first_name, "Updated Name 1") | ||||
|         self.assertEqual(returned_user1.password, initial_password) | ||||
| 
 | ||||
|         response = self.client.patch( | ||||
|             f"{self.ENDPOINT}{user1.pk}/", | ||||
|             data={ | ||||
|                 "first_name": "Updated Name 2", | ||||
|                 "password": "123xyz", | ||||
|             }, | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
| 
 | ||||
|         returned_user2 = User.objects.get(pk=user1.pk) | ||||
|         self.assertEqual(returned_user2.first_name, "Updated Name 2") | ||||
|         self.assertNotEqual(returned_user2.password, initial_password) | ||||
| 
 | ||||
| 
 | ||||
| class TestApiGroup(DirectoriesMixin, APITestCase): | ||||
|     ENDPOINT = "/api/groups/" | ||||
| 
 | ||||
|     def setUp(self): | ||||
|         super().setUp() | ||||
| 
 | ||||
|         self.user = User.objects.create_superuser(username="temp_admin") | ||||
|         self.client.force_authenticate(user=self.user) | ||||
| 
 | ||||
|     def test_get_groups(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Configured groups | ||||
|         WHEN: | ||||
|             - API call is made to get groups | ||||
|         THEN: | ||||
|             - Configured groups are provided | ||||
|         """ | ||||
| 
 | ||||
|         group1 = Group.objects.create( | ||||
|             name="Test Group", | ||||
|         ) | ||||
| 
 | ||||
|         response = self.client.get(self.ENDPOINT) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(response.data["count"], 1) | ||||
|         returned_group1 = response.data["results"][0] | ||||
| 
 | ||||
|         self.assertEqual(returned_group1["name"], group1.name) | ||||
| 
 | ||||
|     def test_create_group(self): | ||||
|         """ | ||||
|         WHEN: | ||||
|             - API request is made to add a group | ||||
|         THEN: | ||||
|             - A new group is created | ||||
|         """ | ||||
| 
 | ||||
|         group1 = { | ||||
|             "name": "Test Group", | ||||
|         } | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             data=group1, | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_201_CREATED) | ||||
| 
 | ||||
|         returned_group1 = Group.objects.get(name="Test Group") | ||||
| 
 | ||||
|         self.assertEqual(returned_group1.name, group1["name"]) | ||||
| 
 | ||||
|     def test_delete_group(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Existing group | ||||
|         WHEN: | ||||
|             - API request is made to delete a group | ||||
|         THEN: | ||||
|             - Group is deleted | ||||
|         """ | ||||
| 
 | ||||
|         group1 = Group.objects.create( | ||||
|             name="Test Group", | ||||
|         ) | ||||
| 
 | ||||
|         response = self.client.delete( | ||||
|             f"{self.ENDPOINT}{group1.pk}/", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_204_NO_CONTENT) | ||||
| 
 | ||||
|         self.assertEqual(len(Group.objects.all()), 0) | ||||
| 
 | ||||
|     def test_update_group(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Existing groups | ||||
|         WHEN: | ||||
|             - API request is made to update group | ||||
|         THEN: | ||||
|             - The group is updated | ||||
|         """ | ||||
| 
 | ||||
|         group1 = Group.objects.create( | ||||
|             name="Test Group", | ||||
|         ) | ||||
| 
 | ||||
|         response = self.client.patch( | ||||
|             f"{self.ENDPOINT}{group1.pk}/", | ||||
|             data={ | ||||
|                 "name": "Updated Name 1", | ||||
|             }, | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
| 
 | ||||
|         returned_group1 = Group.objects.get(pk=group1.pk) | ||||
|         self.assertEqual(returned_group1.name, "Updated Name 1") | ||||
| 
 | ||||
| 
 | ||||
| class TestBulkEditObjectPermissions(APITestCase): | ||||
|     def setUp(self): | ||||
|         super().setUp() | ||||
| 
 | ||||
|         user = User.objects.create_superuser(username="temp_admin") | ||||
|         self.client.force_authenticate(user=user) | ||||
| 
 | ||||
|         self.t1 = Tag.objects.create(name="t1") | ||||
|         self.t2 = Tag.objects.create(name="t2") | ||||
|         self.c1 = Correspondent.objects.create(name="c1") | ||||
|         self.dt1 = DocumentType.objects.create(name="dt1") | ||||
|         self.sp1 = StoragePath.objects.create(name="sp1") | ||||
|         self.user1 = User.objects.create(username="user1") | ||||
|         self.user2 = User.objects.create(username="user2") | ||||
|         self.user3 = User.objects.create(username="user3") | ||||
| 
 | ||||
|     def test_bulk_object_set_permissions(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Existing objects | ||||
|         WHEN: | ||||
|             - bulk_edit_object_perms API endpoint is called | ||||
|         THEN: | ||||
|             - Permissions and / or owner are changed | ||||
|         """ | ||||
|         permissions = { | ||||
|             "view": { | ||||
|                 "users": [self.user1.id, self.user2.id], | ||||
|                 "groups": [], | ||||
|             }, | ||||
|             "change": { | ||||
|                 "users": [self.user1.id], | ||||
|                 "groups": [], | ||||
|             }, | ||||
|         } | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/bulk_edit_object_perms/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "objects": [self.t1.id, self.t2.id], | ||||
|                     "object_type": "tags", | ||||
|                     "permissions": permissions, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertIn(self.user1, get_users_with_perms(self.t1)) | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/bulk_edit_object_perms/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "objects": [self.c1.id], | ||||
|                     "object_type": "correspondents", | ||||
|                     "permissions": permissions, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertIn(self.user1, get_users_with_perms(self.c1)) | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/bulk_edit_object_perms/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "objects": [self.dt1.id], | ||||
|                     "object_type": "document_types", | ||||
|                     "permissions": permissions, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertIn(self.user1, get_users_with_perms(self.dt1)) | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/bulk_edit_object_perms/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "objects": [self.sp1.id], | ||||
|                     "object_type": "storage_paths", | ||||
|                     "permissions": permissions, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertIn(self.user1, get_users_with_perms(self.sp1)) | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/bulk_edit_object_perms/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "objects": [self.t1.id, self.t2.id], | ||||
|                     "object_type": "tags", | ||||
|                     "owner": self.user3.id, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(Tag.objects.get(pk=self.t2.id).owner, self.user3) | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/bulk_edit_object_perms/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "objects": [self.sp1.id], | ||||
|                     "object_type": "storage_paths", | ||||
|                     "owner": self.user3.id, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(StoragePath.objects.get(pk=self.sp1.id).owner, self.user3) | ||||
| 
 | ||||
|     def test_bulk_edit_object_permissions_insufficient_perms(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Objects owned by user other than logged in user | ||||
|         WHEN: | ||||
|             - bulk_edit_object_perms API endpoint is called | ||||
|         THEN: | ||||
|             - User is not able to change permissions | ||||
|         """ | ||||
|         self.t1.owner = User.objects.get(username="temp_admin") | ||||
|         self.t1.save() | ||||
|         self.client.force_authenticate(user=self.user1) | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             "/api/bulk_edit_object_perms/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "objects": [self.t1.id, self.t2.id], | ||||
|                     "object_type": "tags", | ||||
|                     "owner": self.user1.id, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_403_FORBIDDEN) | ||||
|         self.assertEqual(response.content, b"Insufficient permissions") | ||||
| 
 | ||||
|     def test_bulk_edit_object_permissions_validation(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Existing objects | ||||
|         WHEN: | ||||
|             - bulk_edit_object_perms API endpoint is called with invalid params | ||||
|         THEN: | ||||
|             - Validation fails | ||||
|         """ | ||||
|         # not a list | ||||
|         response = self.client.post( | ||||
|             "/api/bulk_edit_object_perms/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "objects": self.t1.id, | ||||
|                     "object_type": "tags", | ||||
|                     "owner": self.user1.id, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
| 
 | ||||
|         # not a list of ints | ||||
|         response = self.client.post( | ||||
|             "/api/bulk_edit_object_perms/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "objects": ["one"], | ||||
|                     "object_type": "tags", | ||||
|                     "owner": self.user1.id, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
| 
 | ||||
|         # duplicates | ||||
|         response = self.client.post( | ||||
|             "/api/bulk_edit_object_perms/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "objects": [self.t1.id, self.t2.id, self.t1.id], | ||||
|                     "object_type": "tags", | ||||
|                     "owner": self.user1.id, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
| 
 | ||||
|         # not a valid object type | ||||
|         response = self.client.post( | ||||
|             "/api/bulk_edit_object_perms/", | ||||
|             json.dumps( | ||||
|                 { | ||||
|                     "objects": [1], | ||||
|                     "object_type": "madeup", | ||||
|                     "owner": self.user1.id, | ||||
|                 }, | ||||
|             ), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) | ||||
							
								
								
									
										123
									
								
								src/documents/tests/test_api_remote_version.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										123
									
								
								src/documents/tests/test_api_remote_version.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,123 @@ | ||||
| import json | ||||
| import urllib.request | ||||
| from unittest import mock | ||||
| from unittest.mock import MagicMock | ||||
| 
 | ||||
| from rest_framework import status | ||||
| from rest_framework.test import APITestCase | ||||
| 
 | ||||
| from documents.tests.utils import DirectoriesMixin | ||||
| from paperless import version | ||||
| 
 | ||||
| 
 | ||||
| class TestApiRemoteVersion(DirectoriesMixin, APITestCase): | ||||
|     ENDPOINT = "/api/remote_version/" | ||||
| 
 | ||||
|     def setUp(self): | ||||
|         super().setUp() | ||||
| 
 | ||||
|     @mock.patch("urllib.request.urlopen") | ||||
|     def test_remote_version_enabled_no_update_prefix(self, urlopen_mock): | ||||
|         cm = MagicMock() | ||||
|         cm.getcode.return_value = status.HTTP_200_OK | ||||
|         cm.read.return_value = json.dumps({"tag_name": "ngx-1.6.0"}).encode() | ||||
|         cm.__enter__.return_value = cm | ||||
|         urlopen_mock.return_value = cm | ||||
| 
 | ||||
|         response = self.client.get(self.ENDPOINT) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertDictEqual( | ||||
|             response.data, | ||||
|             { | ||||
|                 "version": "1.6.0", | ||||
|                 "update_available": False, | ||||
|             }, | ||||
|         ) | ||||
| 
 | ||||
|     @mock.patch("urllib.request.urlopen") | ||||
|     def test_remote_version_enabled_no_update_no_prefix(self, urlopen_mock): | ||||
|         cm = MagicMock() | ||||
|         cm.getcode.return_value = status.HTTP_200_OK | ||||
|         cm.read.return_value = json.dumps( | ||||
|             {"tag_name": version.__full_version_str__}, | ||||
|         ).encode() | ||||
|         cm.__enter__.return_value = cm | ||||
|         urlopen_mock.return_value = cm | ||||
| 
 | ||||
|         response = self.client.get(self.ENDPOINT) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertDictEqual( | ||||
|             response.data, | ||||
|             { | ||||
|                 "version": version.__full_version_str__, | ||||
|                 "update_available": False, | ||||
|             }, | ||||
|         ) | ||||
| 
 | ||||
|     @mock.patch("urllib.request.urlopen") | ||||
|     def test_remote_version_enabled_update(self, urlopen_mock): | ||||
|         new_version = ( | ||||
|             version.__version__[0], | ||||
|             version.__version__[1], | ||||
|             version.__version__[2] + 1, | ||||
|         ) | ||||
|         new_version_str = ".".join(map(str, new_version)) | ||||
| 
 | ||||
|         cm = MagicMock() | ||||
|         cm.getcode.return_value = status.HTTP_200_OK | ||||
|         cm.read.return_value = json.dumps( | ||||
|             {"tag_name": new_version_str}, | ||||
|         ).encode() | ||||
|         cm.__enter__.return_value = cm | ||||
|         urlopen_mock.return_value = cm | ||||
| 
 | ||||
|         response = self.client.get(self.ENDPOINT) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertDictEqual( | ||||
|             response.data, | ||||
|             { | ||||
|                 "version": new_version_str, | ||||
|                 "update_available": True, | ||||
|             }, | ||||
|         ) | ||||
| 
 | ||||
|     @mock.patch("urllib.request.urlopen") | ||||
|     def test_remote_version_bad_json(self, urlopen_mock): | ||||
|         cm = MagicMock() | ||||
|         cm.getcode.return_value = status.HTTP_200_OK | ||||
|         cm.read.return_value = b'{ "blah":' | ||||
|         cm.__enter__.return_value = cm | ||||
|         urlopen_mock.return_value = cm | ||||
| 
 | ||||
|         response = self.client.get(self.ENDPOINT) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertDictEqual( | ||||
|             response.data, | ||||
|             { | ||||
|                 "version": "0.0.0", | ||||
|                 "update_available": False, | ||||
|             }, | ||||
|         ) | ||||
| 
 | ||||
|     @mock.patch("urllib.request.urlopen") | ||||
|     def test_remote_version_exception(self, urlopen_mock): | ||||
|         cm = MagicMock() | ||||
|         cm.getcode.return_value = status.HTTP_200_OK | ||||
|         cm.read.side_effect = urllib.error.URLError("an error") | ||||
|         cm.__enter__.return_value = cm | ||||
|         urlopen_mock.return_value = cm | ||||
| 
 | ||||
|         response = self.client.get(self.ENDPOINT) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertDictEqual( | ||||
|             response.data, | ||||
|             { | ||||
|                 "version": "0.0.0", | ||||
|                 "update_available": False, | ||||
|             }, | ||||
|         ) | ||||
							
								
								
									
										240
									
								
								src/documents/tests/test_api_tasks.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										240
									
								
								src/documents/tests/test_api_tasks.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,240 @@ | ||||
| import uuid | ||||
| 
 | ||||
| import celery | ||||
| from django.contrib.auth.models import User | ||||
| from rest_framework import status | ||||
| from rest_framework.test import APITestCase | ||||
| 
 | ||||
| from documents.models import PaperlessTask | ||||
| from documents.tests.utils import DirectoriesMixin | ||||
| 
 | ||||
| 
 | ||||
| class TestTasks(DirectoriesMixin, APITestCase): | ||||
|     ENDPOINT = "/api/tasks/" | ||||
|     ENDPOINT_ACKNOWLEDGE = "/api/acknowledge_tasks/" | ||||
| 
 | ||||
|     def setUp(self): | ||||
|         super().setUp() | ||||
| 
 | ||||
|         self.user = User.objects.create_superuser(username="temp_admin") | ||||
|         self.client.force_authenticate(user=self.user) | ||||
| 
 | ||||
|     def test_get_tasks(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Attempted celery tasks | ||||
|         WHEN: | ||||
|             - API call is made to get tasks | ||||
|         THEN: | ||||
|             - Attempting and pending tasks are serialized and provided | ||||
|         """ | ||||
| 
 | ||||
|         task1 = PaperlessTask.objects.create( | ||||
|             task_id=str(uuid.uuid4()), | ||||
|             task_file_name="task_one.pdf", | ||||
|         ) | ||||
| 
 | ||||
|         task2 = PaperlessTask.objects.create( | ||||
|             task_id=str(uuid.uuid4()), | ||||
|             task_file_name="task_two.pdf", | ||||
|         ) | ||||
| 
 | ||||
|         response = self.client.get(self.ENDPOINT) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(len(response.data), 2) | ||||
|         returned_task1 = response.data[1] | ||||
|         returned_task2 = response.data[0] | ||||
| 
 | ||||
|         self.assertEqual(returned_task1["task_id"], task1.task_id) | ||||
|         self.assertEqual(returned_task1["status"], celery.states.PENDING) | ||||
|         self.assertEqual(returned_task1["task_file_name"], task1.task_file_name) | ||||
| 
 | ||||
|         self.assertEqual(returned_task2["task_id"], task2.task_id) | ||||
|         self.assertEqual(returned_task2["status"], celery.states.PENDING) | ||||
|         self.assertEqual(returned_task2["task_file_name"], task2.task_file_name) | ||||
| 
 | ||||
|     def test_get_single_task_status(self): | ||||
|         """ | ||||
|         GIVEN | ||||
|             - Query parameter for a valid task ID | ||||
|         WHEN: | ||||
|             - API call is made to get task status | ||||
|         THEN: | ||||
|             - Single task data is returned | ||||
|         """ | ||||
| 
 | ||||
|         id1 = str(uuid.uuid4()) | ||||
|         task1 = PaperlessTask.objects.create( | ||||
|             task_id=id1, | ||||
|             task_file_name="task_one.pdf", | ||||
|         ) | ||||
| 
 | ||||
|         _ = PaperlessTask.objects.create( | ||||
|             task_id=str(uuid.uuid4()), | ||||
|             task_file_name="task_two.pdf", | ||||
|         ) | ||||
| 
 | ||||
|         response = self.client.get(self.ENDPOINT + f"?task_id={id1}") | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(len(response.data), 1) | ||||
|         returned_task1 = response.data[0] | ||||
| 
 | ||||
|         self.assertEqual(returned_task1["task_id"], task1.task_id) | ||||
| 
 | ||||
|     def test_get_single_task_status_not_valid(self): | ||||
|         """ | ||||
|         GIVEN | ||||
|             - Query parameter for a non-existent task ID | ||||
|         WHEN: | ||||
|             - API call is made to get task status | ||||
|         THEN: | ||||
|             - No task data is returned | ||||
|         """ | ||||
|         PaperlessTask.objects.create( | ||||
|             task_id=str(uuid.uuid4()), | ||||
|             task_file_name="task_one.pdf", | ||||
|         ) | ||||
| 
 | ||||
|         _ = PaperlessTask.objects.create( | ||||
|             task_id=str(uuid.uuid4()), | ||||
|             task_file_name="task_two.pdf", | ||||
|         ) | ||||
| 
 | ||||
|         response = self.client.get(self.ENDPOINT + "?task_id=bad-task-id") | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(len(response.data), 0) | ||||
| 
 | ||||
|     def test_acknowledge_tasks(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Attempted celery tasks | ||||
|         WHEN: | ||||
|             - API call is made to get mark task as acknowledged | ||||
|         THEN: | ||||
|             - Task is marked as acknowledged | ||||
|         """ | ||||
|         task = PaperlessTask.objects.create( | ||||
|             task_id=str(uuid.uuid4()), | ||||
|             task_file_name="task_one.pdf", | ||||
|         ) | ||||
| 
 | ||||
|         response = self.client.get(self.ENDPOINT) | ||||
|         self.assertEqual(len(response.data), 1) | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT_ACKNOWLEDGE, | ||||
|             {"tasks": [task.id]}, | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
| 
 | ||||
|         response = self.client.get(self.ENDPOINT) | ||||
|         self.assertEqual(len(response.data), 0) | ||||
| 
 | ||||
|     def test_task_result_no_error(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - A celery task completed without error | ||||
|         WHEN: | ||||
|             - API call is made to get tasks | ||||
|         THEN: | ||||
|             - The returned data includes the task result | ||||
|         """ | ||||
|         PaperlessTask.objects.create( | ||||
|             task_id=str(uuid.uuid4()), | ||||
|             task_file_name="task_one.pdf", | ||||
|             status=celery.states.SUCCESS, | ||||
|             result="Success. New document id 1 created", | ||||
|         ) | ||||
| 
 | ||||
|         response = self.client.get(self.ENDPOINT) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(len(response.data), 1) | ||||
| 
 | ||||
|         returned_data = response.data[0] | ||||
| 
 | ||||
|         self.assertEqual(returned_data["result"], "Success. New document id 1 created") | ||||
|         self.assertEqual(returned_data["related_document"], "1") | ||||
| 
 | ||||
|     def test_task_result_with_error(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - A celery task completed with an exception | ||||
|         WHEN: | ||||
|             - API call is made to get tasks | ||||
|         THEN: | ||||
|             - The returned result is the exception info | ||||
|         """ | ||||
|         PaperlessTask.objects.create( | ||||
|             task_id=str(uuid.uuid4()), | ||||
|             task_file_name="task_one.pdf", | ||||
|             status=celery.states.FAILURE, | ||||
|             result="test.pdf: Not consuming test.pdf: It is a duplicate.", | ||||
|         ) | ||||
| 
 | ||||
|         response = self.client.get(self.ENDPOINT) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(len(response.data), 1) | ||||
| 
 | ||||
|         returned_data = response.data[0] | ||||
| 
 | ||||
|         self.assertEqual( | ||||
|             returned_data["result"], | ||||
|             "test.pdf: Not consuming test.pdf: It is a duplicate.", | ||||
|         ) | ||||
| 
 | ||||
|     def test_task_name_webui(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Attempted celery task | ||||
|             - Task was created through the webui | ||||
|         WHEN: | ||||
|             - API call is made to get tasks | ||||
|         THEN: | ||||
|             - Returned data include the filename | ||||
|         """ | ||||
|         PaperlessTask.objects.create( | ||||
|             task_id=str(uuid.uuid4()), | ||||
|             task_file_name="test.pdf", | ||||
|             task_name="documents.tasks.some_task", | ||||
|             status=celery.states.SUCCESS, | ||||
|         ) | ||||
| 
 | ||||
|         response = self.client.get(self.ENDPOINT) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(len(response.data), 1) | ||||
| 
 | ||||
|         returned_data = response.data[0] | ||||
| 
 | ||||
|         self.assertEqual(returned_data["task_file_name"], "test.pdf") | ||||
| 
 | ||||
|     def test_task_name_consume_folder(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - Attempted celery task | ||||
|             - Task was created through the consume folder | ||||
|         WHEN: | ||||
|             - API call is made to get tasks | ||||
|         THEN: | ||||
|             - Returned data include the filename | ||||
|         """ | ||||
|         PaperlessTask.objects.create( | ||||
|             task_id=str(uuid.uuid4()), | ||||
|             task_file_name="anothertest.pdf", | ||||
|             task_name="documents.tasks.some_task", | ||||
|             status=celery.states.SUCCESS, | ||||
|         ) | ||||
| 
 | ||||
|         response = self.client.get(self.ENDPOINT) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertEqual(len(response.data), 1) | ||||
| 
 | ||||
|         returned_data = response.data[0] | ||||
| 
 | ||||
|         self.assertEqual(returned_data["task_file_name"], "anothertest.pdf") | ||||
							
								
								
									
										65
									
								
								src/documents/tests/test_api_uisettings.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								src/documents/tests/test_api_uisettings.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,65 @@ | ||||
| import json | ||||
| 
 | ||||
| from django.contrib.auth.models import User | ||||
| from rest_framework import status | ||||
| from rest_framework.test import APITestCase | ||||
| 
 | ||||
| from documents.tests.utils import DirectoriesMixin | ||||
| 
 | ||||
| 
 | ||||
| class TestApiUiSettings(DirectoriesMixin, APITestCase): | ||||
|     ENDPOINT = "/api/ui_settings/" | ||||
| 
 | ||||
|     def setUp(self): | ||||
|         super().setUp() | ||||
|         self.test_user = User.objects.create_superuser(username="test") | ||||
|         self.test_user.first_name = "Test" | ||||
|         self.test_user.last_name = "User" | ||||
|         self.test_user.save() | ||||
|         self.client.force_authenticate(user=self.test_user) | ||||
| 
 | ||||
|     def test_api_get_ui_settings(self): | ||||
|         response = self.client.get(self.ENDPOINT, format="json") | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|         self.assertDictEqual( | ||||
|             response.data["user"], | ||||
|             { | ||||
|                 "id": self.test_user.id, | ||||
|                 "username": self.test_user.username, | ||||
|                 "is_superuser": True, | ||||
|                 "groups": [], | ||||
|                 "first_name": self.test_user.first_name, | ||||
|                 "last_name": self.test_user.last_name, | ||||
|             }, | ||||
|         ) | ||||
|         self.assertDictEqual( | ||||
|             response.data["settings"], | ||||
|             { | ||||
|                 "update_checking": { | ||||
|                     "backend_setting": "default", | ||||
|                 }, | ||||
|             }, | ||||
|         ) | ||||
| 
 | ||||
|     def test_api_set_ui_settings(self): | ||||
|         settings = { | ||||
|             "settings": { | ||||
|                 "dark_mode": { | ||||
|                     "enabled": True, | ||||
|                 }, | ||||
|             }, | ||||
|         } | ||||
| 
 | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT, | ||||
|             json.dumps(settings), | ||||
|             content_type="application/json", | ||||
|         ) | ||||
| 
 | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
| 
 | ||||
|         ui_settings = self.test_user.ui_settings | ||||
|         self.assertDictEqual( | ||||
|             ui_settings.settings, | ||||
|             settings["settings"], | ||||
|         ) | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user