From a95383cfa2605e752d9b36a37bf1f75db8d6757f Mon Sep 17 00:00:00 2001 From: Thu Trang Pham Date: Sun, 8 Nov 2020 09:51:49 -0800 Subject: [PATCH] Formatted with black and reached 100% coverage --- Makefile | 9 + README.md | 2 + admin_confirm/admin.py | 60 ++-- admin_confirm/tests/test_admin.py | 282 ++++++++++++++---- coverage.svg | 21 ++ setup.py | 24 +- tests/factories.py | 5 +- tests/manage.py | 4 +- tests/market/admin.py | 8 +- tests/market/apps.py | 2 +- tests/market/migrations/0001_initial.py | 49 ++- .../migrations/0002_auto_20201031_2057.py | 55 +++- .../migrations/0003_auto_20201108_1717.py | 25 ++ tests/market/models.py | 16 +- tests/test_project/settings.py | 78 +++-- tests/test_project/urls.py | 2 +- tests/test_project/wsgi.py | 2 +- 17 files changed, 465 insertions(+), 179 deletions(-) create mode 100644 coverage.svg create mode 100644 tests/market/migrations/0003_auto_20201108_1717.py diff --git a/Makefile b/Makefile index bf35ab4..7f5d621 100644 --- a/Makefile +++ b/Makefile @@ -1,2 +1,11 @@ run: ./tests/manage.py runserver + +test: + coverage run --branch -m pytest + coverage html + coverage-badge -f -o coverage.svg + +migrate: + ./tests/manage.py makemigrations + ./tests/manage.py migrate diff --git a/README.md b/README.md index 36564a5..6ea9dec 100644 --- a/README.md +++ b/README.md @@ -1,5 +1,7 @@ # Django Admin Confirm +![coverage](/coverage.svg) + AdminConfirmMixin is a mixin for ModelAdmin to add confirmations to changes and additions. ![Screenshot of Confirmation Page](/screenshot.png) diff --git a/admin_confirm/admin.py b/admin_confirm/admin.py index 58539df..c70ada6 100644 --- a/admin_confirm/admin.py +++ b/admin_confirm/admin.py @@ -52,21 +52,23 @@ class AdminConfirmMixin: def changeform_view(self, request, object_id=None, form_url="", extra_context=None): if request.method == "POST": - if (not object_id and "_confirm_add" in request.POST) or (object_id and "_confirm_change" in request.POST): - return self._change_confirmation_view(request, object_id, form_url, extra_context) + if (not object_id and "_confirm_add" in request.POST) or ( + object_id and "_confirm_change" in request.POST + ): + return self._change_confirmation_view( + request, object_id, form_url, extra_context + ) extra_context = { **(extra_context or {}), - 'confirm_add': self.confirm_add, - 'confirm_change': self.confirm_change + "confirm_add": self.confirm_add, + "confirm_change": self.confirm_change, } return super().changeform_view(request, object_id, form_url, extra_context) def _change_confirmation_view(self, request, object_id, form_url, extra_context): # This code is taken from super()._changeform_view - to_field = request.POST.get( - TO_FIELD_VAR, request.GET.get(TO_FIELD_VAR) - ) + to_field = request.POST.get(TO_FIELD_VAR, request.GET.get(TO_FIELD_VAR)) if to_field and not self.to_field_allowed(request, to_field): raise DisallowedModelAdminToField( "The field %s cannot be referenced." % to_field @@ -83,7 +85,6 @@ class AdminConfirmMixin: obj = None else: obj = self.get_object(request, unquote(object_id), to_field) - if obj is None: return self._get_obj_does_not_exist_redirect(request, opts, object_id) @@ -105,21 +106,30 @@ class AdminConfirmMixin: # End code from super()._changeform_view changed_data = {} - if add: - for name in form.changed_data: - new_value = new_object.__getattribute__(name) - if new_value is not None: - changed_data[name] = [None, new_value] - else: - # Parse the changed data - Note that using form.changed_data would not work because initial is not set - for name, field in form.fields.items(): - initial_value = obj.__getattribute__(name) - new_value = new_object.__getattribute__(name) - if field.has_changed(initial_value, new_value) and initial_value != new_value: - changed_data[name] = [initial_value, new_value] + if form_validated: + if add: + for name in form.changed_data: + new_value = getattr(new_object, name) + # Don't consider default values as changed for adding + if ( + new_value is not None + and new_value != model._meta.get_field(name).default + ): + changed_data[name] = [None, new_value] + else: + # Parse the changed data - Note that using form.changed_data would not work because initial is not set + for name, field in form.fields.items(): + initial_value = getattr(obj, name) + new_value = getattr(new_object, name) + if ( + field.has_changed(initial_value, new_value) + and initial_value != new_value + ): + changed_data[name] = [initial_value, new_value] - changed_confirmation_fields = set(self.get_confirmation_fields( - request, obj)) & set(changed_data.keys()) + changed_confirmation_fields = set( + self.get_confirmation_fields(request, obj) + ) & set(changed_data.keys()) if not bool(changed_confirmation_fields): # No confirmation required for changed fields, continue to save return super()._changeform_view(request, object_id, form_url, extra_context) @@ -132,14 +142,14 @@ class AdminConfirmMixin: if key in ["_save", "_saveasnew", "_addanother", "_continue"]: save_action = key - if key.startswith("_") or key == 'csrfmiddlewaretoken': + if key.startswith("_") or key == "csrfmiddlewaretoken": continue form_data[key] = request.POST.get(key) if add: - title_action = _('adding') + title_action = _("adding") else: - title_action = _('changing') + title_action = _("changing") context = { **self.admin_site.each_context(request), diff --git a/admin_confirm/tests/test_admin.py b/admin_confirm/tests/test_admin.py index c71c0b0..5ece1b4 100644 --- a/admin_confirm/tests/test_admin.py +++ b/admin_confirm/tests/test_admin.py @@ -1,135 +1,303 @@ from django.test import TestCase, RequestFactory from django.contrib.auth.models import User from django.contrib.admin.sites import AdminSite - -from tests.market.admin import ItemAdmin -from tests.market.models import Item, Inventory +from django.contrib.admin.options import TO_FIELD_VAR +from django.http import HttpResponseForbidden, HttpResponseBadRequest from django.urls import reverse -from tests.factories import ItemFactory, ShopFactory + + +from tests.market.admin import ItemAdmin, InventoryAdmin +from tests.market.models import Item, Inventory +from tests.factories import ItemFactory, ShopFactory, InventoryFactory class TestAdminConfirmMixin(TestCase): @classmethod def setUpTestData(cls): cls.superuser = User.objects.create_superuser( - username='super', email='super@email.org', password='pass') + username="super", email="super@email.org", password="pass" + ) def setUp(self): self.client.force_login(self.superuser) self.factory = RequestFactory() def test_get_add_without_confirm_add(self): - response = self.client.get(reverse('admin:market_item_add')) - self.assertFalse(response.context_data.get('confirm_add')) - self.assertNotIn('_confirm_add', response.rendered_content) + response = self.client.get(reverse("admin:market_item_add")) + self.assertFalse(response.context_data.get("confirm_add")) + self.assertNotIn("_confirm_add", response.rendered_content) def test_get_add_with_confirm_add(self): - response = self.client.get(reverse('admin:market_inventory_add')) - self.assertTrue(response.context_data.get('confirm_add')) - self.assertIn('_confirm_add', response.rendered_content) + response = self.client.get(reverse("admin:market_inventory_add")) + self.assertTrue(response.context_data.get("confirm_add")) + self.assertIn("_confirm_add", response.rendered_content) def test_get_change_without_confirm_change(self): - response = self.client.get(reverse('admin:market_shop_add')) - self.assertFalse(response.context_data.get('confirm_change')) - self.assertNotIn('_confirm_change', response.rendered_content) + response = self.client.get(reverse("admin:market_shop_add")) + self.assertFalse(response.context_data.get("confirm_change")) + self.assertNotIn("_confirm_change", response.rendered_content) def test_get_change_with_confirm_change(self): - response = self.client.get(reverse('admin:market_inventory_add')) - self.assertTrue(response.context_data.get('confirm_change')) - self.assertIn('_confirm_change', response.rendered_content) + response = self.client.get(reverse("admin:market_inventory_add")) + self.assertTrue(response.context_data.get("confirm_change")) + self.assertIn("_confirm_change", response.rendered_content) def test_post_add_without_confirm_add(self): - data = {'name': 'name', 'price': 2.0, - 'currency': Item.VALID_CURRENCIES[0]} - response = self.client.post(reverse('admin:market_item_add'), data) + data = {"name": "name", "price": 2.0, "currency": Item.VALID_CURRENCIES[0]} + response = self.client.post(reverse("admin:market_item_add"), data) # Redirects to item changelist and item is added self.assertEqual(response.status_code, 302) - self.assertEqual(response.url, '/admin/market/item/') + self.assertEqual(response.url, "/admin/market/item/") self.assertEqual(Item.objects.count(), 1) def test_post_add_with_confirm_add(self): item = ItemFactory() shop = ShopFactory() - data = {'shop': shop.id, 'item': item.id, - 'quantity': 5, '_confirm_add': True} - response = self.client.post( - reverse('admin:market_inventory_add'), data) + data = {"shop": shop.id, "item": item.id, "quantity": 5, "_confirm_add": True} + response = self.client.post(reverse("admin:market_inventory_add"), data) # Ensure not redirected (confirmation page does not redirect) self.assertEqual(response.status_code, 200) expected_templates = [ - 'admin/market/inventory/change_confirmation.html', - 'admin/market/change_confirmation.html', - 'admin/change_confirmation.html' + "admin/market/inventory/change_confirmation.html", + "admin/market/change_confirmation.html", + "admin/change_confirmation.html", ] self.assertEqual(response.template_name, expected_templates) - form_data = {'shop': str(shop.id), 'item': str( - item.id), 'quantity': str(5)} - self.assertEqual( - response.context_data['form_data'], form_data) + form_data = {"shop": str(shop.id), "item": str(item.id), "quantity": str(5)} + self.assertEqual(response.context_data["form_data"], form_data) for k, v in form_data.items(): self.assertIn( - f'', response.rendered_content) + f'', + response.rendered_content, + ) # Should not have been added yet self.assertEqual(Inventory.objects.count(), 0) def test_post_change_with_confirm_change(self): - item = ItemFactory(name='item') - data = {'name': 'name', 'price': 2.0, - 'currency': Item.VALID_CURRENCIES[0], '_confirm_change': True} - response = self.client.post( - f'/admin/market/item/{item.id}/change/', data) + item = ItemFactory(name="item") + data = { + "name": "name", + "price": 2.0, + "currency": Item.VALID_CURRENCIES[0], + "id": item.id, + "_confirm_change": True, + "csrfmiddlewaretoken": "fake token", + "_save": True, + } + response = self.client.post(f"/admin/market/item/{item.id}/change/", data) # Ensure not redirected (confirmation page does not redirect) self.assertEqual(response.status_code, 200) expected_templates = [ - 'admin/market/item/change_confirmation.html', - 'admin/market/change_confirmation.html', - 'admin/change_confirmation.html' + "admin/market/item/change_confirmation.html", + "admin/market/change_confirmation.html", + "admin/change_confirmation.html", ] self.assertEqual(response.template_name, expected_templates) - form_data = {'name': 'name', 'price': str(2.0), - 'currency': Item.VALID_CURRENCIES[0][0]} - self.assertEqual( - response.context_data['form_data'], form_data) + form_data = { + "name": "name", + "price": str(2.0), + "id": str(item.id), + "currency": Item.VALID_CURRENCIES[0][0], + } + self.assertEqual(response.context_data["form_data"], form_data) for k, v in form_data.items(): self.assertIn( - f'', response.rendered_content) + f'', + response.rendered_content, + ) # Hasn't changed item yet item.refresh_from_db() - self.assertEqual(item.name, 'item') + self.assertEqual(item.name, "item") def test_post_change_without_confirm_change(self): - shop = ShopFactory(name='bob') - data = {'name': 'sally'} - response = self.client.post( - f'/admin/market/shop/{shop.id}/change/', data) + shop = ShopFactory(name="bob") + data = {"name": "sally"} + response = self.client.post(f"/admin/market/shop/{shop.id}/change/", data) # Redirects to changelist self.assertEqual(response.status_code, 302) - self.assertEqual(response.url, '/admin/market/shop/') + self.assertEqual(response.url, "/admin/market/shop/") # Shop has changed shop.refresh_from_db() - self.assertEqual(shop.name, 'sally') + self.assertEqual(shop.name, "sally") def test_get_confirmation_fields_should_default_if_not_set(self): - expected_fields = [f.name for f in Item._meta.fields if f.name != 'id'] + expected_fields = [f.name for f in Item._meta.fields if f.name != "id"] ItemAdmin.confirmation_fields = None admin = ItemAdmin(Item, AdminSite()) actual_fields = admin.get_confirmation_fields(self.factory.request()) self.assertEqual(expected_fields, actual_fields) def test_get_confirmation_fields_if_set(self): - expected_fields = ['name', 'currency'] + expected_fields = ["name", "currency"] ItemAdmin.confirmation_fields = expected_fields admin = ItemAdmin(Item, AdminSite()) actual_fields = admin.get_confirmation_fields(self.factory.request()) self.assertEqual(expected_fields, actual_fields) def test_custom_template(self): - expected_template = 'market/admin/my_custom_template.html' + expected_template = "market/admin/my_custom_template.html" ItemAdmin.confirmation_template = expected_template admin = ItemAdmin(Item, AdminSite()) actual_template = admin.render_change_confirmation( - self.factory.request(), context={}).template_name + self.factory.request(), context={} + ).template_name self.assertEqual(expected_template, actual_template) ItemAdmin.confirmation_template = None + + def test_form_invalid(self): + self.assertEqual(InventoryAdmin.confirmation_fields, ["quantity"]) + + inventory = InventoryFactory(quantity=1) + data = { + "quantity": 1, + "shop": "Invalid value", + "item": "Invalid value", + "id": inventory.id, + "_confirm_change": True, + "csrfmiddlewaretoken": "fake token", + } + response = self.client.post( + f"/admin/market/inventory/{inventory.id}/change/", data + ) + + # Form invalid should show erros on form + self.assertEqual(response.status_code, 200) + print(response.rendered_content) + self.assertIsNotNone(response.context_data.get("errors")) + self.assertEqual( + response.context_data["errors"][0], + ["Select a valid choice. That choice is not one of the available choices."], + ) + # Should not have updated inventory + inventory.refresh_from_db() + self.assertEqual(inventory.quantity, 1) + + def test_confirmation_fields_set_with_confirm_change(self): + self.assertEqual(InventoryAdmin.confirmation_fields, ["quantity"]) + + inventory = InventoryFactory() + another_shop = ShopFactory() + data = { + "quantity": inventory.quantity, + "id": inventory.id, + "item": inventory.item.id, + "shop": another_shop.id, + "_confirm_change": True, + "csrfmiddlewaretoken": "fake token", + } + response = self.client.post( + f"/admin/market/inventory/{inventory.id}/change/", data + ) + + # Should not have shown confirmation page since shop did not change + self.assertEqual(response.status_code, 302) + self.assertEqual(response.url, reverse("admin:market_inventory_changelist")) + # Should have updated inventory + inventory.refresh_from_db() + self.assertEqual(inventory.shop, another_shop) + + def test_confirmation_fields_set_with_confirm_add(self): + self.assertEqual(InventoryAdmin.confirmation_fields, ["quantity"]) + + item = ItemFactory() + shop = ShopFactory() + + # Don't set quantity - let it default + data = {"shop": shop.id, "item": item.id, "_confirm_add": True} + response = self.client.post(reverse("admin:market_inventory_add"), data) + # No confirmation needed + self.assertEqual(response.status_code, 302) + + # Should have been added + self.assertEqual(Inventory.objects.count(), 1) + new_inventory = Inventory.objects.all().first() + self.assertEqual(new_inventory.shop, shop) + self.assertEqual(new_inventory.item, item) + self.assertEqual( + new_inventory.quantity, Inventory._meta.get_field("quantity").default + ) + + def test_no_change_permissions(self): + user = User.objects.create_user(username="user", is_staff=True) + self.client.force_login(user) + + inventory = InventoryFactory() + data = { + "quantity": 1000, + "id": inventory.id, + "item": inventory.item.id, + "shop": inventory.shop.id, + "_confirm_change": True, + "csrfmiddlewaretoken": "fake token", + } + response = self.client.post( + f"/admin/market/inventory/{inventory.id}/change/", data + ) + + self.assertEqual(response.status_code, 403) + self.assertTrue(isinstance(response, HttpResponseForbidden)) + + old_quantity = inventory.quantity + inventory.refresh_from_db() + self.assertEqual(inventory.quantity, old_quantity) + + def test_no_add_permissions(self): + user = User.objects.create_user(username="user", is_staff=True) + self.client.force_login(user) + item = ItemFactory() + shop = ShopFactory() + data = {"shop": shop.id, "item": item.id, "quantity": 5, "_confirm_add": True} + response = self.client.post(reverse("admin:market_inventory_add"), data) + # Ensure not redirected (confirmation page does not redirect) + self.assertEqual(response.status_code, 403) + self.assertTrue(isinstance(response, HttpResponseForbidden)) + + # Should not have been added + self.assertEqual(Inventory.objects.count(), 0) + + def test_obj_not_found(self): + inventory = InventoryFactory() + data = { + "quantity": 1000, + "id": 100, + "item": inventory.item.id, + "shop": inventory.shop.id, + "_confirm_change": True, + "csrfmiddlewaretoken": "fake token", + } + response = self.client.post("/admin/market/inventory/100/change/", data) + + self.assertEqual(response.status_code, 302) + self.assertEqual(response.url, "/admin/") + self.assertEqual(response.reason_phrase, "Found") + + old_quantity = inventory.quantity + inventory.refresh_from_db() + self.assertEqual(inventory.quantity, old_quantity) + + self.assertEqual(Inventory.objects.count(), 1) + + def test_handles_to_field_not_allowed(self): + item = ItemFactory() + shop = ShopFactory() + data = { + "shop": shop.id, + "item": item.id, + "quantity": 5, + "_confirm_add": True, + TO_FIELD_VAR: "shop", + } + response = self.client.post(reverse("admin:market_inventory_add"), data) + # Ensure not redirected (confirmation page does not redirect) + self.assertEqual(response.status_code, 400) + self.assertTrue(isinstance(response, HttpResponseBadRequest)) + self.assertEqual(response.reason_phrase, "Bad Request") + self.assertEqual( + response.context.get("exception_value"), + "The field shop cannot be referenced.", + ) + + # Should not have been added + self.assertEqual(Inventory.objects.count(), 0) diff --git a/coverage.svg b/coverage.svg new file mode 100644 index 0000000..e5db27c --- /dev/null +++ b/coverage.svg @@ -0,0 +1,21 @@ + + + + + + + + + + + + + + + + coverage + coverage + 100% + 100% + + diff --git a/setup.py b/setup.py index 284f506..d7c88d0 100644 --- a/setup.py +++ b/setup.py @@ -2,19 +2,19 @@ import os from setuptools import setup here = os.path.abspath(os.path.dirname(__file__)) -README = open(os.path.join(here, 'README.md')).read() +README = open(os.path.join(here, "README.md")).read() setup( - name='django-admin-confirm', - version='0.1', - packages=['admin_confirm'], - description='Adds confirmation to Django Admin changes and additions', + name="django-admin-confirm", + version="0.1", + packages=["admin_confirm"], + description="Adds confirmation to Django Admin changes and additions", long_description=README, - author='Thu Trang Pham', - author_email='thuutrangpham@gmail.com', - url='https://github.com/trangpham/django-admin-confirm/', - license='Apache 2.0', + author="Thu Trang Pham", + author_email="thuutrangpham@gmail.com", + url="https://github.com/trangpham/django-admin-confirm/", + license="Apache 2.0", install_requires=[ - 'Django>=1.7', - ] -) \ No newline at end of file + "Django>=1.7", + ], +) diff --git a/tests/factories.py b/tests/factories.py index 924d89c..9180118 100644 --- a/tests/factories.py +++ b/tests/factories.py @@ -9,7 +9,7 @@ class ItemFactory(factory.django.DjangoModelFactory): class Meta: model = Item - name = factory.Faker('name') + name = factory.Faker("name") price = factory.LazyAttribute(lambda _: randint(5, 500)) currency = factory.LazyAttribute(lambda _: choice(Item.VALID_CURRENCIES)) @@ -18,7 +18,7 @@ class ShopFactory(factory.django.DjangoModelFactory): class Meta: model = Shop - name = factory.Faker('name') + name = factory.Faker("name") class InventoryFactory(factory.django.DjangoModelFactory): @@ -28,4 +28,3 @@ class InventoryFactory(factory.django.DjangoModelFactory): shop = factory.SubFactory(ShopFactory) item = factory.SubFactory(ItemFactory) quantity = factory.Sequence(lambda n: n) - diff --git a/tests/manage.py b/tests/manage.py index 2753b50..4d788bb 100755 --- a/tests/manage.py +++ b/tests/manage.py @@ -5,7 +5,7 @@ import sys def main(): - os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'test_project.settings') + os.environ.setdefault("DJANGO_SETTINGS_MODULE", "test_project.settings") try: from django.core.management import execute_from_command_line except ImportError as exc: @@ -17,5 +17,5 @@ def main(): execute_from_command_line(sys.argv) -if __name__ == '__main__': +if __name__ == "__main__": main() diff --git a/tests/market/admin.py b/tests/market/admin.py index 9362698..100c65f 100644 --- a/tests/market/admin.py +++ b/tests/market/admin.py @@ -6,19 +6,19 @@ from .models import Item, Inventory, Shop class ItemAdmin(AdminConfirmMixin, admin.ModelAdmin): - list_display = ('name', 'price', 'currency') + list_display = ("name", "price", "currency") confirm_change = True class InventoryAdmin(AdminConfirmMixin, admin.ModelAdmin): - list_display = ('shop', 'item', 'quantity') + list_display = ("shop", "item", "quantity") confirm_change = True confirm_add = True - confirmation_fields = ['shop'] + confirmation_fields = ["quantity"] class ShopAdmin(AdminConfirmMixin, admin.ModelAdmin): - confirmation_fields = ['name'] + confirmation_fields = ["name"] admin.site.register(Item, ItemAdmin) diff --git a/tests/market/apps.py b/tests/market/apps.py index f9a1da8..0d79fee 100644 --- a/tests/market/apps.py +++ b/tests/market/apps.py @@ -2,4 +2,4 @@ from django.apps import AppConfig class MarketConfig(AppConfig): - name = 'market' + name = "market" diff --git a/tests/market/migrations/0001_initial.py b/tests/market/migrations/0001_initial.py index 943fbbf..af69340 100644 --- a/tests/market/migrations/0001_initial.py +++ b/tests/market/migrations/0001_initial.py @@ -8,25 +8,52 @@ class Migration(migrations.Migration): initial = True - dependencies = [ - ] + dependencies = [] operations = [ migrations.CreateModel( - name='Item', + name="Item", fields=[ - ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), - ('name', models.CharField(max_length=120)), - ('price', models.DecimalField(decimal_places=2, max_digits=5)), - ('currency', models.CharField(choices=[('CAD', 'CAD'), ('USD', 'USD')], max_length=3)), + ( + "id", + models.AutoField( + auto_created=True, + primary_key=True, + serialize=False, + verbose_name="ID", + ), + ), + ("name", models.CharField(max_length=120)), + ("price", models.DecimalField(decimal_places=2, max_digits=5)), + ( + "currency", + models.CharField( + choices=[("CAD", "CAD"), ("USD", "USD")], max_length=3 + ), + ), ], ), migrations.CreateModel( - name='Stock', + name="Stock", fields=[ - ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), - ('quantity', models.PositiveIntegerField()), - ('item', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='all_stock', to='market.Item')), + ( + "id", + models.AutoField( + auto_created=True, + primary_key=True, + serialize=False, + verbose_name="ID", + ), + ), + ("quantity", models.PositiveIntegerField()), + ( + "item", + models.ForeignKey( + on_delete=django.db.models.deletion.CASCADE, + related_name="all_stock", + to="market.Item", + ), + ), ], ), ] diff --git a/tests/market/migrations/0002_auto_20201031_2057.py b/tests/market/migrations/0002_auto_20201031_2057.py index 184e6bd..8d88caf 100644 --- a/tests/market/migrations/0002_auto_20201031_2057.py +++ b/tests/market/migrations/0002_auto_20201031_2057.py @@ -7,38 +7,63 @@ import django.db.models.deletion class Migration(migrations.Migration): dependencies = [ - ('market', '0001_initial'), + ("market", "0001_initial"), ] operations = [ migrations.CreateModel( - name='Inventory', + name="Inventory", fields=[ - ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), - ('quantity', models.PositiveIntegerField()), - ('item', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='market.Item')), + ( + "id", + models.AutoField( + auto_created=True, + primary_key=True, + serialize=False, + verbose_name="ID", + ), + ), + ("quantity", models.PositiveIntegerField()), + ( + "item", + models.ForeignKey( + on_delete=django.db.models.deletion.CASCADE, to="market.Item" + ), + ), ], options={ - 'ordering': ['shop', 'item__name'], + "ordering": ["shop", "item__name"], }, ), migrations.CreateModel( - name='Shop', + name="Shop", fields=[ - ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), - ('name', models.CharField(max_length=120)), + ( + "id", + models.AutoField( + auto_created=True, + primary_key=True, + serialize=False, + verbose_name="ID", + ), + ), + ("name", models.CharField(max_length=120)), ], ), migrations.DeleteModel( - name='Stock', + name="Stock", ), migrations.AddField( - model_name='inventory', - name='shop', - field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='inventory', to='market.Shop'), + model_name="inventory", + name="shop", + field=models.ForeignKey( + on_delete=django.db.models.deletion.CASCADE, + related_name="inventory", + to="market.Shop", + ), ), migrations.AlterUniqueTogether( - name='inventory', - unique_together={('shop', 'item')}, + name="inventory", + unique_together={("shop", "item")}, ), ] diff --git a/tests/market/migrations/0003_auto_20201108_1717.py b/tests/market/migrations/0003_auto_20201108_1717.py new file mode 100644 index 0000000..17e87ca --- /dev/null +++ b/tests/market/migrations/0003_auto_20201108_1717.py @@ -0,0 +1,25 @@ +# Generated by Django 3.0.10 on 2020-11-08 17:17 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ("market", "0002_auto_20201031_2057"), + ] + + operations = [ + migrations.AlterModelOptions( + name="inventory", + options={ + "ordering": ["shop", "item__name"], + "verbose_name_plural": "Inventory", + }, + ), + migrations.AlterField( + model_name="inventory", + name="quantity", + field=models.PositiveIntegerField(blank=True, default=0, null=True), + ), + ] diff --git a/tests/market/models.py b/tests/market/models.py index c279178..8b8a0bd 100644 --- a/tests/market/models.py +++ b/tests/market/models.py @@ -3,8 +3,8 @@ from django.db import models class Item(models.Model): VALID_CURRENCIES = ( - ('CAD', 'CAD'), - ('USD', 'USD'), + ("CAD", "CAD"), + ("USD", "USD"), ) name = models.CharField(max_length=120) price = models.DecimalField(max_digits=5, decimal_places=2) @@ -23,10 +23,12 @@ class Shop(models.Model): class Inventory(models.Model): class Meta: - unique_together = ['shop', 'item'] - ordering = ['shop', 'item__name'] - verbose_name_plural = 'Inventory' + unique_together = ["shop", "item"] + ordering = ["shop", "item__name"] + verbose_name_plural = "Inventory" - shop = models.ForeignKey(to=Shop, on_delete=models.CASCADE, related_name='inventory') + shop = models.ForeignKey( + to=Shop, on_delete=models.CASCADE, related_name="inventory" + ) item = models.ForeignKey(to=Item, on_delete=models.CASCADE) - quantity = models.PositiveIntegerField() + quantity = models.PositiveIntegerField(default=0, null=True, blank=True) diff --git a/tests/test_project/settings.py b/tests/test_project/settings.py index 385918e..629218e 100644 --- a/tests/test_project/settings.py +++ b/tests/test_project/settings.py @@ -19,67 +19,65 @@ BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) # See https://docs.djangoproject.com/en/3.0/howto/deployment/checklist/ # SECURITY WARNING: keep the secret key used in production secret! -SECRET_KEY = '=yddl-40388w3e2hl$e8)revce=n67_idi8pfejtn3!+2%!_qt' +SECRET_KEY = "=yddl-40388w3e2hl$e8)revce=n67_idi8pfejtn3!+2%!_qt" # SECURITY WARNING: don't run with debug turned on in production! DEBUG = True -ALLOWED_HOSTS = ['127.0.0.1'] +ALLOWED_HOSTS = ["127.0.0.1"] # Application definition INSTALLED_APPS = [ - 'admin_confirm', - - 'django.contrib.admin', - 'django.contrib.auth', - 'django.contrib.contenttypes', - 'django.contrib.sessions', - 'django.contrib.messages', - 'django.contrib.staticfiles', - - 'tests.market', + "admin_confirm", + "django.contrib.admin", + "django.contrib.auth", + "django.contrib.contenttypes", + "django.contrib.sessions", + "django.contrib.messages", + "django.contrib.staticfiles", + "tests.market", ] MIDDLEWARE = [ - 'django.middleware.security.SecurityMiddleware', - 'django.contrib.sessions.middleware.SessionMiddleware', - 'django.middleware.common.CommonMiddleware', - 'django.middleware.csrf.CsrfViewMiddleware', - 'django.contrib.auth.middleware.AuthenticationMiddleware', - 'django.contrib.messages.middleware.MessageMiddleware', - 'django.middleware.clickjacking.XFrameOptionsMiddleware', + "django.middleware.security.SecurityMiddleware", + "django.contrib.sessions.middleware.SessionMiddleware", + "django.middleware.common.CommonMiddleware", + "django.middleware.csrf.CsrfViewMiddleware", + "django.contrib.auth.middleware.AuthenticationMiddleware", + "django.contrib.messages.middleware.MessageMiddleware", + "django.middleware.clickjacking.XFrameOptionsMiddleware", ] -ROOT_URLCONF = 'tests.test_project.urls' +ROOT_URLCONF = "tests.test_project.urls" TEMPLATES = [ { - 'BACKEND': 'django.template.backends.django.DjangoTemplates', - 'DIRS': [], - 'APP_DIRS': True, - 'OPTIONS': { - 'context_processors': [ - 'django.template.context_processors.debug', - 'django.template.context_processors.request', - 'django.contrib.auth.context_processors.auth', - 'django.contrib.messages.context_processors.messages', + "BACKEND": "django.template.backends.django.DjangoTemplates", + "DIRS": [], + "APP_DIRS": True, + "OPTIONS": { + "context_processors": [ + "django.template.context_processors.debug", + "django.template.context_processors.request", + "django.contrib.auth.context_processors.auth", + "django.contrib.messages.context_processors.messages", ], }, }, ] -WSGI_APPLICATION = 'tests.test_project.wsgi.application' +WSGI_APPLICATION = "tests.test_project.wsgi.application" # Database # https://docs.djangoproject.com/en/3.0/ref/settings/#databases DATABASES = { - 'default': { - 'ENGINE': 'django.db.backends.sqlite3', - 'NAME': os.path.join(BASE_DIR, 'db.sqlite3'), + "default": { + "ENGINE": "django.db.backends.sqlite3", + "NAME": os.path.join(BASE_DIR, "db.sqlite3"), } } @@ -89,16 +87,16 @@ DATABASES = { AUTH_PASSWORD_VALIDATORS = [ { - 'NAME': 'django.contrib.auth.password_validation.UserAttributeSimilarityValidator', + "NAME": "django.contrib.auth.password_validation.UserAttributeSimilarityValidator", }, { - 'NAME': 'django.contrib.auth.password_validation.MinimumLengthValidator', + "NAME": "django.contrib.auth.password_validation.MinimumLengthValidator", }, { - 'NAME': 'django.contrib.auth.password_validation.CommonPasswordValidator', + "NAME": "django.contrib.auth.password_validation.CommonPasswordValidator", }, { - 'NAME': 'django.contrib.auth.password_validation.NumericPasswordValidator', + "NAME": "django.contrib.auth.password_validation.NumericPasswordValidator", }, ] @@ -106,9 +104,9 @@ AUTH_PASSWORD_VALIDATORS = [ # Internationalization # https://docs.djangoproject.com/en/3.0/topics/i18n/ -LANGUAGE_CODE = 'en-us' +LANGUAGE_CODE = "en-us" -TIME_ZONE = 'UTC' +TIME_ZONE = "UTC" USE_I18N = True @@ -120,4 +118,4 @@ USE_TZ = True # Static files (CSS, JavaScript, Images) # https://docs.djangoproject.com/en/3.0/howto/static-files/ -STATIC_URL = '/static/' +STATIC_URL = "/static/" diff --git a/tests/test_project/urls.py b/tests/test_project/urls.py index dfc7362..083932c 100644 --- a/tests/test_project/urls.py +++ b/tests/test_project/urls.py @@ -2,5 +2,5 @@ from django.contrib import admin from django.urls import path urlpatterns = [ - path('admin/', admin.site.urls), + path("admin/", admin.site.urls), ] diff --git a/tests/test_project/wsgi.py b/tests/test_project/wsgi.py index 1e6481c..535131b 100644 --- a/tests/test_project/wsgi.py +++ b/tests/test_project/wsgi.py @@ -2,6 +2,6 @@ import os from django.core.wsgi import get_wsgi_application -os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'test_project.settings') +os.environ.setdefault("DJANGO_SETTINGS_MODULE", "test_project.settings") application = get_wsgi_application()