wbcore 1.59.9__py2.py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- wbcore/__init__.py +1 -0
- wbcore/admin.py +197 -0
- wbcore/apps.py +17 -0
- wbcore/cache/__init__.py +0 -0
- wbcore/cache/buttons.py +23 -0
- wbcore/cache/decorators.py +31 -0
- wbcore/cache/mixins.py +49 -0
- wbcore/cache/registry.py +98 -0
- wbcore/cache/views.py +19 -0
- wbcore/configs/__init__.py +11 -0
- wbcore/configs/configs.py +52 -0
- wbcore/configs/decorators.py +11 -0
- wbcore/configs/registry.py +35 -0
- wbcore/configs/views.py +12 -0
- wbcore/configurations/__init__.py +1 -0
- wbcore/configurations/base.py +46 -0
- wbcore/configurations/configurations/__init__.py +16 -0
- wbcore/configurations/configurations/apps.py +55 -0
- wbcore/configurations/configurations/authentication.py +44 -0
- wbcore/configurations/configurations/base.py +15 -0
- wbcore/configurations/configurations/cache.py +20 -0
- wbcore/configurations/configurations/celery.py +26 -0
- wbcore/configurations/configurations/i18nl10n.py +11 -0
- wbcore/configurations/configurations/mail.py +2 -0
- wbcore/configurations/configurations/maintenance.py +53 -0
- wbcore/configurations/configurations/media.py +22 -0
- wbcore/configurations/configurations/middleware.py +27 -0
- wbcore/configurations/configurations/network.py +19 -0
- wbcore/configurations/configurations/rest_framework.py +42 -0
- wbcore/configurations/configurations/static.py +28 -0
- wbcore/configurations/configurations/templates.py +17 -0
- wbcore/configurations/configurations/uvicorn.py +9 -0
- wbcore/configurations/configurations/wbcore.py +61 -0
- wbcore/content_type/__init__.py +0 -0
- wbcore/content_type/admin.py +8 -0
- wbcore/content_type/filters.py +20 -0
- wbcore/content_type/serializers.py +98 -0
- wbcore/content_type/utils.py +30 -0
- wbcore/content_type/viewsets.py +81 -0
- wbcore/contrib/__init__.py +0 -0
- wbcore/contrib/agenda/__init__.py +0 -0
- wbcore/contrib/agenda/admin/__init__.py +2 -0
- wbcore/contrib/agenda/admin/calendar_item.py +14 -0
- wbcore/contrib/agenda/admin/conference_room.py +19 -0
- wbcore/contrib/agenda/apps.py +6 -0
- wbcore/contrib/agenda/configurations.py +11 -0
- wbcore/contrib/agenda/factories/__init__.py +2 -0
- wbcore/contrib/agenda/factories/calendar_item.py +49 -0
- wbcore/contrib/agenda/factories/conference_room.py +20 -0
- wbcore/contrib/agenda/filters/__init__.py +2 -0
- wbcore/contrib/agenda/filters/calendar_item.py +66 -0
- wbcore/contrib/agenda/filters/conference_room.py +42 -0
- wbcore/contrib/agenda/locale/de/LC_MESSAGES/django.po +206 -0
- wbcore/contrib/agenda/locale/de/LC_MESSAGES/django.po.translated +236 -0
- wbcore/contrib/agenda/locale/en/LC_MESSAGES/django.po +200 -0
- wbcore/contrib/agenda/locale/fr/LC_MESSAGES/django.po +201 -0
- wbcore/contrib/agenda/migrations/0001_initial.py +84 -0
- wbcore/contrib/agenda/migrations/0002_initial.py +26 -0
- wbcore/contrib/agenda/migrations/0003_calendaritem_endpoint_basename.py +42 -0
- wbcore/contrib/agenda/migrations/0004_alter_calendaritem_item_type.py +17 -0
- wbcore/contrib/agenda/migrations/0005_building_and_more.py +94 -0
- wbcore/contrib/agenda/migrations/0006_calendaritem_is_deletable.py +17 -0
- wbcore/contrib/agenda/migrations/0007_alter_calendaritem_options.py +21 -0
- wbcore/contrib/agenda/migrations/0008_alter_calendaritem_item_type.py +17 -0
- wbcore/contrib/agenda/migrations/0009_alter_calendaritem_icon.py +18 -0
- wbcore/contrib/agenda/migrations/__init__.py +0 -0
- wbcore/contrib/agenda/models/__init__.py +2 -0
- wbcore/contrib/agenda/models/calendar_item.py +240 -0
- wbcore/contrib/agenda/models/conference_room.py +96 -0
- wbcore/contrib/agenda/release_notes/1_0_0.md +13 -0
- wbcore/contrib/agenda/release_notes/__init__.py +0 -0
- wbcore/contrib/agenda/serializers/__init__.py +10 -0
- wbcore/contrib/agenda/serializers/calendar_item.py +77 -0
- wbcore/contrib/agenda/serializers/conference_room.py +99 -0
- wbcore/contrib/agenda/signals.py +5 -0
- wbcore/contrib/agenda/static/agenda/markdown/documentation/building.md +11 -0
- wbcore/contrib/agenda/static/agenda/markdown/documentation/conferenceroom.md +20 -0
- wbcore/contrib/agenda/tests/__init__.py +0 -0
- wbcore/contrib/agenda/tests/conftest.py +14 -0
- wbcore/contrib/agenda/tests/signals.py +17 -0
- wbcore/contrib/agenda/tests/test_models.py +34 -0
- wbcore/contrib/agenda/tests/test_viewsets.py +172 -0
- wbcore/contrib/agenda/tests/tests.py +26 -0
- wbcore/contrib/agenda/typings.py +19 -0
- wbcore/contrib/agenda/urls.py +26 -0
- wbcore/contrib/agenda/viewsets/__init__.py +13 -0
- wbcore/contrib/agenda/viewsets/buttons/__init__.py +1 -0
- wbcore/contrib/agenda/viewsets/buttons/conference_room.py +21 -0
- wbcore/contrib/agenda/viewsets/calendar_items.py +169 -0
- wbcore/contrib/agenda/viewsets/conference_room.py +48 -0
- wbcore/contrib/agenda/viewsets/display/__init__.py +2 -0
- wbcore/contrib/agenda/viewsets/display/calendar_items.py +17 -0
- wbcore/contrib/agenda/viewsets/display/conference_room.py +42 -0
- wbcore/contrib/agenda/viewsets/endpoints/__init__.py +1 -0
- wbcore/contrib/agenda/viewsets/endpoints/calendar_items.py +19 -0
- wbcore/contrib/agenda/viewsets/menu/__init__.py +2 -0
- wbcore/contrib/agenda/viewsets/menu/calendar_items.py +12 -0
- wbcore/contrib/agenda/viewsets/menu/conference_room.py +38 -0
- wbcore/contrib/agenda/viewsets/titles/__init__.py +2 -0
- wbcore/contrib/agenda/viewsets/titles/calendar_items.py +8 -0
- wbcore/contrib/agenda/viewsets/titles/conference_room.py +25 -0
- wbcore/contrib/ai/__init__.py +0 -0
- wbcore/contrib/ai/apps.py +5 -0
- wbcore/contrib/ai/exceptions.py +40 -0
- wbcore/contrib/ai/llm/__init__.py +0 -0
- wbcore/contrib/ai/llm/config.py +166 -0
- wbcore/contrib/ai/llm/decorators.py +10 -0
- wbcore/contrib/ai/llm/mixins.py +34 -0
- wbcore/contrib/ai/llm/utils.py +69 -0
- wbcore/contrib/authentication/__init__.py +9 -0
- wbcore/contrib/authentication/admin.py +247 -0
- wbcore/contrib/authentication/apps.py +14 -0
- wbcore/contrib/authentication/authentication.py +111 -0
- wbcore/contrib/authentication/configs.py +10 -0
- wbcore/contrib/authentication/configurations.py +60 -0
- wbcore/contrib/authentication/dynamic_preferences_registry.py +26 -0
- wbcore/contrib/authentication/factories/__init__.py +10 -0
- wbcore/contrib/authentication/factories/tokens.py +15 -0
- wbcore/contrib/authentication/factories/users.py +112 -0
- wbcore/contrib/authentication/factories/users_activities.py +19 -0
- wbcore/contrib/authentication/filters.py +17 -0
- wbcore/contrib/authentication/fixtures/authentication.json +62 -0
- wbcore/contrib/authentication/locale/de/LC_MESSAGES/django.po +632 -0
- wbcore/contrib/authentication/locale/de/LC_MESSAGES/django.po.translated +634 -0
- wbcore/contrib/authentication/locale/en/LC_MESSAGES/django.po +590 -0
- wbcore/contrib/authentication/locale/fr/LC_MESSAGES/django.po +592 -0
- wbcore/contrib/authentication/management/__init__.py +14 -0
- wbcore/contrib/authentication/migrations/0001_initial_squashed.py +174 -0
- wbcore/contrib/authentication/migrations/0002_profile.py +26 -0
- wbcore/contrib/authentication/migrations/0003_alter_user_profile.py +34 -0
- wbcore/contrib/authentication/migrations/0004_token.py +51 -0
- wbcore/contrib/authentication/migrations/0005_user_external_calendar_settings.py +17 -0
- wbcore/contrib/authentication/migrations/0006_auto_20231206_1422.py +13 -0
- wbcore/contrib/authentication/migrations/__init__.py +0 -0
- wbcore/contrib/authentication/models/__init__.py +3 -0
- wbcore/contrib/authentication/models/tokens.py +140 -0
- wbcore/contrib/authentication/models/users.py +224 -0
- wbcore/contrib/authentication/models/users_activities.py +114 -0
- wbcore/contrib/authentication/release_notes/1_0_0.md +13 -0
- wbcore/contrib/authentication/release_notes/__init__.py +0 -0
- wbcore/contrib/authentication/serializers/__init__.py +14 -0
- wbcore/contrib/authentication/serializers/users.py +351 -0
- wbcore/contrib/authentication/serializers/users_activites.py +37 -0
- wbcore/contrib/authentication/tasks.py +30 -0
- wbcore/contrib/authentication/templates/activate_confirm.html +12 -0
- wbcore/contrib/authentication/templates/base.html +135 -0
- wbcore/contrib/authentication/templates/email_base_template.html +335 -0
- wbcore/contrib/authentication/templates/password_reset_done.html +13 -0
- wbcore/contrib/authentication/templates/password_reset_email.html +11 -0
- wbcore/contrib/authentication/templates/password_reset_email_html.html +43 -0
- wbcore/contrib/authentication/templates/password_reset_form.html +23 -0
- wbcore/contrib/authentication/templates/password_reset_sent.html +11 -0
- wbcore/contrib/authentication/templates/reset_password.html +15 -0
- wbcore/contrib/authentication/templates/user_registration_email.html +37 -0
- wbcore/contrib/authentication/tests/__init__.py +0 -0
- wbcore/contrib/authentication/tests/conftest.py +18 -0
- wbcore/contrib/authentication/tests/e2e/__init__.py +1 -0
- wbcore/contrib/authentication/tests/e2e/e2e_auth_utility.py +20 -0
- wbcore/contrib/authentication/tests/signals.py +18 -0
- wbcore/contrib/authentication/tests/test_configs.py +6 -0
- wbcore/contrib/authentication/tests/test_serializers.py +6 -0
- wbcore/contrib/authentication/tests/test_tasks.py +34 -0
- wbcore/contrib/authentication/tests/test_tokens.py +127 -0
- wbcore/contrib/authentication/tests/test_users.py +338 -0
- wbcore/contrib/authentication/tests/test_viewsets.py +6 -0
- wbcore/contrib/authentication/tests/tests.py +15 -0
- wbcore/contrib/authentication/urls.py +94 -0
- wbcore/contrib/authentication/utils.py +10 -0
- wbcore/contrib/authentication/viewsets/__init__.py +19 -0
- wbcore/contrib/authentication/viewsets/buttons/__init__.py +1 -0
- wbcore/contrib/authentication/viewsets/buttons/users.py +56 -0
- wbcore/contrib/authentication/viewsets/display/__init__.py +6 -0
- wbcore/contrib/authentication/viewsets/display/user_activities.py +86 -0
- wbcore/contrib/authentication/viewsets/display/users.py +73 -0
- wbcore/contrib/authentication/viewsets/endpoints/__init__.py +6 -0
- wbcore/contrib/authentication/viewsets/endpoints/user_activities.py +14 -0
- wbcore/contrib/authentication/viewsets/endpoints/users.py +18 -0
- wbcore/contrib/authentication/viewsets/menu/__init__.py +6 -0
- wbcore/contrib/authentication/viewsets/menu/user_activities.py +40 -0
- wbcore/contrib/authentication/viewsets/menu/users.py +17 -0
- wbcore/contrib/authentication/viewsets/titles/__init__.py +12 -0
- wbcore/contrib/authentication/viewsets/titles/user_activities.py +32 -0
- wbcore/contrib/authentication/viewsets/titles/users.py +27 -0
- wbcore/contrib/authentication/viewsets/user_activities.py +224 -0
- wbcore/contrib/authentication/viewsets/users.py +331 -0
- wbcore/contrib/color/admin.py +28 -0
- wbcore/contrib/color/apps.py +5 -0
- wbcore/contrib/color/enums.py +17 -0
- wbcore/contrib/color/factories.py +10 -0
- wbcore/contrib/color/fields.py +29 -0
- wbcore/contrib/color/forms.py +13 -0
- wbcore/contrib/color/migrations/0001_initial.py +33 -0
- wbcore/contrib/color/migrations/0002_alter_colorgradient_colors.py +25 -0
- wbcore/contrib/color/migrations/__init__.py +0 -0
- wbcore/contrib/color/models.py +63 -0
- wbcore/contrib/color/tests/conftest.py +10 -0
- wbcore/contrib/color/tests/test_color_models.py +61 -0
- wbcore/contrib/color/tests/test_fields.py +25 -0
- wbcore/contrib/currency/__init__.py +0 -0
- wbcore/contrib/currency/admin.py +33 -0
- wbcore/contrib/currency/apps.py +5 -0
- wbcore/contrib/currency/dynamic_preferences_registry.py +39 -0
- wbcore/contrib/currency/factories.py +40 -0
- wbcore/contrib/currency/fixtures/currency.yaml +1014 -0
- wbcore/contrib/currency/fixtures/currency_fx_rate.yaml +73585 -0
- wbcore/contrib/currency/import_export/__init__.py +0 -0
- wbcore/contrib/currency/import_export/backends/__init__.py +1 -0
- wbcore/contrib/currency/import_export/backends/fixerio/__init__.py +1 -0
- wbcore/contrib/currency/import_export/backends/fixerio/currency_fx_rates.py +69 -0
- wbcore/contrib/currency/import_export/backends/utils.py +5 -0
- wbcore/contrib/currency/import_export/handlers/__init__.py +2 -0
- wbcore/contrib/currency/import_export/handlers/currency.py +25 -0
- wbcore/contrib/currency/import_export/handlers/currency_fx_rates.py +29 -0
- wbcore/contrib/currency/import_export/parsers/__init__.py +0 -0
- wbcore/contrib/currency/import_export/parsers/fixerio/__init__.py +0 -0
- wbcore/contrib/currency/import_export/parsers/fixerio/currency_fx_rates.py +34 -0
- wbcore/contrib/currency/migrations/0001_initial.py +89 -0
- wbcore/contrib/currency/migrations/__init__.py +0 -0
- wbcore/contrib/currency/models.py +199 -0
- wbcore/contrib/currency/release_notes/1_0_0.md +13 -0
- wbcore/contrib/currency/release_notes/__init__.py +0 -0
- wbcore/contrib/currency/serializers.py +45 -0
- wbcore/contrib/currency/tests/__init__.py +0 -0
- wbcore/contrib/currency/tests/conftest.py +7 -0
- wbcore/contrib/currency/tests/test_models.py +115 -0
- wbcore/contrib/currency/tests/test_serializers.py +67 -0
- wbcore/contrib/currency/tests/test_viewsets.py +196 -0
- wbcore/contrib/currency/urls.py +20 -0
- wbcore/contrib/currency/viewsets/__init__.py +2 -0
- wbcore/contrib/currency/viewsets/buttons/__init__.py +0 -0
- wbcore/contrib/currency/viewsets/currency.py +57 -0
- wbcore/contrib/currency/viewsets/currency_fx_rates.py +33 -0
- wbcore/contrib/currency/viewsets/display/__init__.py +2 -0
- wbcore/contrib/currency/viewsets/display/currency.py +38 -0
- wbcore/contrib/currency/viewsets/display/currency_fx_rates.py +9 -0
- wbcore/contrib/currency/viewsets/endpoints/__init__.py +1 -0
- wbcore/contrib/currency/viewsets/endpoints/currency_fx_rates.py +6 -0
- wbcore/contrib/currency/viewsets/menu/__init__.py +1 -0
- wbcore/contrib/currency/viewsets/menu/currency.py +8 -0
- wbcore/contrib/currency/viewsets/preview/__init__.py +1 -0
- wbcore/contrib/currency/viewsets/preview/currency.py +10 -0
- wbcore/contrib/currency/viewsets/titles/__init__.py +2 -0
- wbcore/contrib/currency/viewsets/titles/currency.py +6 -0
- wbcore/contrib/currency/viewsets/titles/currency_fx_rates.py +8 -0
- wbcore/contrib/dataloader/__init__.py +1 -0
- wbcore/contrib/dataloader/apps.py +5 -0
- wbcore/contrib/dataloader/dataloaders/__init__.py +2 -0
- wbcore/contrib/dataloader/dataloaders/dataloaders.py +25 -0
- wbcore/contrib/dataloader/dataloaders/proxies.py +82 -0
- wbcore/contrib/dataloader/models/__init__.py +1 -0
- wbcore/contrib/dataloader/models/entities.py +30 -0
- wbcore/contrib/dataloader/models/querysets/__init__.py +1 -0
- wbcore/contrib/dataloader/models/querysets/entities.py +28 -0
- wbcore/contrib/dataloader/tests/__init__.py +0 -0
- wbcore/contrib/dataloader/tests/conftest.py +26 -0
- wbcore/contrib/dataloader/tests/test/__init__.py +0 -0
- wbcore/contrib/dataloader/tests/test/dataloaders/__init__.py +0 -0
- wbcore/contrib/dataloader/tests/test/dataloaders/dataloaders.py +18 -0
- wbcore/contrib/dataloader/tests/test/dataloaders/protocols.py +5 -0
- wbcore/contrib/dataloader/tests/test/dataloaders/proxies.py +11 -0
- wbcore/contrib/dataloader/tests/test/factories.py +13 -0
- wbcore/contrib/dataloader/tests/test/models.py +11 -0
- wbcore/contrib/dataloader/tests/test_dataloaders.py +38 -0
- wbcore/contrib/dataloader/tests/test_entities.py +13 -0
- wbcore/contrib/dataloader/utils.py +20 -0
- wbcore/contrib/directory/__init__.py +0 -0
- wbcore/contrib/directory/admin/__init__.py +14 -0
- wbcore/contrib/directory/admin/contacts.py +55 -0
- wbcore/contrib/directory/admin/entries.py +235 -0
- wbcore/contrib/directory/admin/relationships.py +73 -0
- wbcore/contrib/directory/apps.py +6 -0
- wbcore/contrib/directory/configs.py +9 -0
- wbcore/contrib/directory/configurations.py +70 -0
- wbcore/contrib/directory/dynamic_preferences_registry.py +61 -0
- wbcore/contrib/directory/factories/__init__.py +35 -0
- wbcore/contrib/directory/factories/contacts.py +71 -0
- wbcore/contrib/directory/factories/entries.py +174 -0
- wbcore/contrib/directory/factories/relationships.py +77 -0
- wbcore/contrib/directory/filters/__init__.py +22 -0
- wbcore/contrib/directory/filters/contacts.py +234 -0
- wbcore/contrib/directory/filters/entries.py +273 -0
- wbcore/contrib/directory/filters/relationships.py +90 -0
- wbcore/contrib/directory/fixtures/directory.json +3884 -0
- wbcore/contrib/directory/locale/de/LC_MESSAGES/django.po +1700 -0
- wbcore/contrib/directory/locale/de/LC_MESSAGES/django.po.translated +1779 -0
- wbcore/contrib/directory/locale/en/LC_MESSAGES/django.po +1652 -0
- wbcore/contrib/directory/locale/fr/LC_MESSAGES/django.po +1654 -0
- wbcore/contrib/directory/migrations/0001_initial.py +871 -0
- wbcore/contrib/directory/migrations/0002_auto_20230414_1553.py +42 -0
- wbcore/contrib/directory/migrations/0003_remove_entry_last_event.py +16 -0
- wbcore/contrib/directory/migrations/0004_entry_is_draft_entry.py +21 -0
- wbcore/contrib/directory/migrations/0005_entry_salutation.py +17 -0
- wbcore/contrib/directory/migrations/0006_employeremployeerelationship_position_name.py +23 -0
- wbcore/contrib/directory/migrations/0007_alter_bankingcontact_options.py +20 -0
- wbcore/contrib/directory/migrations/0008_bankingcontact_access.py +17 -0
- wbcore/contrib/directory/migrations/0009_remove_entry_external_identfier_and_more.py +22 -0
- wbcore/contrib/directory/migrations/0010_remove_addresscontact_city.py +31 -0
- wbcore/contrib/directory/migrations/0011_person_description_person_i18n.py +24 -0
- wbcore/contrib/directory/migrations/0012_alter_person_managers.py +20 -0
- wbcore/contrib/directory/migrations/0013_alter_clientmanagerrelationship_options.py +17 -0
- wbcore/contrib/directory/migrations/0014_alter_entry_relationship_managers_and_more.py +28 -0
- wbcore/contrib/directory/migrations/__init__.py +0 -0
- wbcore/contrib/directory/models/__init__.py +27 -0
- wbcore/contrib/directory/models/contacts.py +554 -0
- wbcore/contrib/directory/models/entries.py +896 -0
- wbcore/contrib/directory/models/relationships.py +624 -0
- wbcore/contrib/directory/permissions.py +6 -0
- wbcore/contrib/directory/preferences.py +11 -0
- wbcore/contrib/directory/release_notes/1_0_0.md +13 -0
- wbcore/contrib/directory/release_notes/1_0_1.md +13 -0
- wbcore/contrib/directory/release_notes/__init__.py +0 -0
- wbcore/contrib/directory/serializers/__init__.py +62 -0
- wbcore/contrib/directory/serializers/companies.py +169 -0
- wbcore/contrib/directory/serializers/contacts.py +404 -0
- wbcore/contrib/directory/serializers/entries.py +355 -0
- wbcore/contrib/directory/serializers/entry_representations.py +38 -0
- wbcore/contrib/directory/serializers/persons.py +214 -0
- wbcore/contrib/directory/serializers/relationships.py +333 -0
- wbcore/contrib/directory/signals.py +3 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/address.md +38 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/banking.md +54 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/bankingentry.md +38 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/clientmanagerrelationship.md +42 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/company.md +52 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/companytype.md +2 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/customerstatus.md +2 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/email.md +20 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/employeecompany.md +34 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/employerperson.md +43 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/person.md +61 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/position.md +2 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/relationshiptype.md +2 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/socialmedia.md +23 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/specialization.md +2 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/systememployee.md +31 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/telephone.md +23 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/telephonesearch.md +26 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/userisclient.md +14 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/userismanager.md +28 -0
- wbcore/contrib/directory/static/directory/markdown/documentation/website.md +20 -0
- wbcore/contrib/directory/tests/__init__.py +0 -0
- wbcore/contrib/directory/tests/conftest.py +62 -0
- wbcore/contrib/directory/tests/disable_signals.py +62 -0
- wbcore/contrib/directory/tests/e2e/__init__.py +7 -0
- wbcore/contrib/directory/tests/e2e/e2e_directory_utility.py +163 -0
- wbcore/contrib/directory/tests/signals.py +89 -0
- wbcore/contrib/directory/tests/test_configs.py +6 -0
- wbcore/contrib/directory/tests/test_filters.py +60 -0
- wbcore/contrib/directory/tests/test_models.py +428 -0
- wbcore/contrib/directory/tests/test_permissions.py +123 -0
- wbcore/contrib/directory/tests/test_serializers.py +217 -0
- wbcore/contrib/directory/tests/test_viewsets.py +835 -0
- wbcore/contrib/directory/typings.py +17 -0
- wbcore/contrib/directory/urls.py +135 -0
- wbcore/contrib/directory/viewsets/__init__.py +57 -0
- wbcore/contrib/directory/viewsets/buttons/__init__.py +7 -0
- wbcore/contrib/directory/viewsets/buttons/contacts.py +17 -0
- wbcore/contrib/directory/viewsets/buttons/entries.py +134 -0
- wbcore/contrib/directory/viewsets/buttons/relationships.py +64 -0
- wbcore/contrib/directory/viewsets/contacts.py +392 -0
- wbcore/contrib/directory/viewsets/display/__init__.py +36 -0
- wbcore/contrib/directory/viewsets/display/contacts.py +368 -0
- wbcore/contrib/directory/viewsets/display/entries.py +532 -0
- wbcore/contrib/directory/viewsets/display/relationships.py +292 -0
- wbcore/contrib/directory/viewsets/display/utils.py +34 -0
- wbcore/contrib/directory/viewsets/endpoints/__init__.py +22 -0
- wbcore/contrib/directory/viewsets/endpoints/contacts.py +55 -0
- wbcore/contrib/directory/viewsets/endpoints/entries.py +31 -0
- wbcore/contrib/directory/viewsets/endpoints/relationships.py +81 -0
- wbcore/contrib/directory/viewsets/entries.py +205 -0
- wbcore/contrib/directory/viewsets/menu/__init__.py +16 -0
- wbcore/contrib/directory/viewsets/menu/contacts.py +26 -0
- wbcore/contrib/directory/viewsets/menu/entries.py +62 -0
- wbcore/contrib/directory/viewsets/menu/relationships.py +32 -0
- wbcore/contrib/directory/viewsets/menu/utils.py +73 -0
- wbcore/contrib/directory/viewsets/mixins.py +12 -0
- wbcore/contrib/directory/viewsets/previews/__init__.py +2 -0
- wbcore/contrib/directory/viewsets/previews/contacts.py +19 -0
- wbcore/contrib/directory/viewsets/previews/entries.py +32 -0
- wbcore/contrib/directory/viewsets/relationships.py +256 -0
- wbcore/contrib/directory/viewsets/titles/__init__.py +30 -0
- wbcore/contrib/directory/viewsets/titles/contacts.py +123 -0
- wbcore/contrib/directory/viewsets/titles/entries.py +30 -0
- wbcore/contrib/directory/viewsets/titles/relationships.py +86 -0
- wbcore/contrib/directory/viewsets/titles/utils.py +47 -0
- wbcore/contrib/directory/viewsets/utils.py +101 -0
- wbcore/contrib/documents/__init__.py +0 -0
- wbcore/contrib/documents/admin.py +80 -0
- wbcore/contrib/documents/apps.py +6 -0
- wbcore/contrib/documents/factories.py +44 -0
- wbcore/contrib/documents/filters.py +81 -0
- wbcore/contrib/documents/fixtures/docments.json +135 -0
- wbcore/contrib/documents/locale/de/LC_MESSAGES/django.po +281 -0
- wbcore/contrib/documents/locale/de/LC_MESSAGES/django.po.translated +285 -0
- wbcore/contrib/documents/locale/en/LC_MESSAGES/django.po +271 -0
- wbcore/contrib/documents/locale/fr/LC_MESSAGES/django.po +270 -0
- wbcore/contrib/documents/migrations/0001_initial.py +189 -0
- wbcore/contrib/documents/migrations/0002_documentmodelrelationship_primary_and_more.py +22 -0
- wbcore/contrib/documents/migrations/0003_alter_documentmodelrelationship_unique_together_and_more.py +30 -0
- wbcore/contrib/documents/migrations/0004_auto_20240103_0958.py +44 -0
- wbcore/contrib/documents/migrations/0005_document_valid_from_document_valid_until_and_more.py +32 -0
- wbcore/contrib/documents/migrations/__init__.py +0 -0
- wbcore/contrib/documents/models/__init__.py +4 -0
- wbcore/contrib/documents/models/document_model_relationships.py +58 -0
- wbcore/contrib/documents/models/document_types.py +36 -0
- wbcore/contrib/documents/models/documents.py +311 -0
- wbcore/contrib/documents/models/mixins.py +10 -0
- wbcore/contrib/documents/models/shareable_links.py +91 -0
- wbcore/contrib/documents/release_notes/1_0_0.md +13 -0
- wbcore/contrib/documents/release_notes/__init__.py +0 -0
- wbcore/contrib/documents/serializers/__init__.py +12 -0
- wbcore/contrib/documents/serializers/document_model_relationships.py +31 -0
- wbcore/contrib/documents/serializers/document_types.py +41 -0
- wbcore/contrib/documents/serializers/documents.py +65 -0
- wbcore/contrib/documents/serializers/shareable_links.py +91 -0
- wbcore/contrib/documents/static/documents/markdown/documentation/document_types.md +21 -0
- wbcore/contrib/documents/static/documents/markdown/documentation/documents.md +18 -0
- wbcore/contrib/documents/static/documents/markdown/documentation/shareablelink.md +28 -0
- wbcore/contrib/documents/static/documents/markdown/documentation/shareablelinkaccess.md +20 -0
- wbcore/contrib/documents/tests/conftest.py +30 -0
- wbcore/contrib/documents/tests/test_models.py +148 -0
- wbcore/contrib/documents/urls.py +43 -0
- wbcore/contrib/documents/viewsets/__init__.py +10 -0
- wbcore/contrib/documents/viewsets/buttons/__init__.py +3 -0
- wbcore/contrib/documents/viewsets/buttons/documents.py +54 -0
- wbcore/contrib/documents/viewsets/buttons/shareable_links.py +21 -0
- wbcore/contrib/documents/viewsets/buttons/signals.py +14 -0
- wbcore/contrib/documents/viewsets/display/__init__.py +4 -0
- wbcore/contrib/documents/viewsets/display/document_model_relationships.py +19 -0
- wbcore/contrib/documents/viewsets/display/document_types.py +24 -0
- wbcore/contrib/documents/viewsets/display/documents.py +102 -0
- wbcore/contrib/documents/viewsets/display/shareable_links.py +66 -0
- wbcore/contrib/documents/viewsets/document_model_relationships.py +25 -0
- wbcore/contrib/documents/viewsets/document_types.py +38 -0
- wbcore/contrib/documents/viewsets/documents.py +108 -0
- wbcore/contrib/documents/viewsets/endpoints/__init__.py +5 -0
- wbcore/contrib/documents/viewsets/endpoints/documents.py +20 -0
- wbcore/contrib/documents/viewsets/endpoints/documents_model_relationships.py +14 -0
- wbcore/contrib/documents/viewsets/endpoints/shareable_links.py +27 -0
- wbcore/contrib/documents/viewsets/menu/__init__.py +1 -0
- wbcore/contrib/documents/viewsets/menu/documents.py +25 -0
- wbcore/contrib/documents/viewsets/previews/__init__.py +1 -0
- wbcore/contrib/documents/viewsets/previews/documents.py +10 -0
- wbcore/contrib/documents/viewsets/shareable_links.py +110 -0
- wbcore/contrib/documents/viewsets/titles/__init__.py +3 -0
- wbcore/contrib/documents/viewsets/titles/document_types.py +14 -0
- wbcore/contrib/documents/viewsets/titles/documents.py +27 -0
- wbcore/contrib/documents/viewsets/titles/shareable_links.py +14 -0
- wbcore/contrib/dynamic_preferences/__init__.py +0 -0
- wbcore/contrib/dynamic_preferences/types.py +127 -0
- wbcore/contrib/dynamic_preferences/viewsets.py +27 -0
- wbcore/contrib/example_app/__init__.py +0 -0
- wbcore/contrib/example_app/admin.py +118 -0
- wbcore/contrib/example_app/apps.py +6 -0
- wbcore/contrib/example_app/factories/__init__.py +8 -0
- wbcore/contrib/example_app/factories/event.py +27 -0
- wbcore/contrib/example_app/factories/league.py +16 -0
- wbcore/contrib/example_app/factories/match.py +26 -0
- wbcore/contrib/example_app/factories/person.py +34 -0
- wbcore/contrib/example_app/factories/role.py +10 -0
- wbcore/contrib/example_app/factories/sport.py +13 -0
- wbcore/contrib/example_app/factories/stadium.py +13 -0
- wbcore/contrib/example_app/factories/team.py +37 -0
- wbcore/contrib/example_app/filters/__init__.py +9 -0
- wbcore/contrib/example_app/filters/event.py +83 -0
- wbcore/contrib/example_app/filters/league.py +75 -0
- wbcore/contrib/example_app/filters/match.py +93 -0
- wbcore/contrib/example_app/filters/person.py +78 -0
- wbcore/contrib/example_app/filters/role.py +18 -0
- wbcore/contrib/example_app/filters/sport.py +27 -0
- wbcore/contrib/example_app/filters/stadium.py +31 -0
- wbcore/contrib/example_app/filters/team.py +78 -0
- wbcore/contrib/example_app/filters/teamresult.py +48 -0
- wbcore/contrib/example_app/fixtures/example_app.json +7425 -0
- wbcore/contrib/example_app/migrations/0001_initial.py +498 -0
- wbcore/contrib/example_app/migrations/0002_sportperson_profile.py +49 -0
- wbcore/contrib/example_app/migrations/0003_change_stadium_capacity.py +31 -0
- wbcore/contrib/example_app/migrations/0004_alter_player_transfer_value.py +21 -0
- wbcore/contrib/example_app/migrations/0005_sportperson_profile_image.py +23 -0
- wbcore/contrib/example_app/migrations/0006_league_season_period_player_is_active_and_more.py +116 -0
- wbcore/contrib/example_app/migrations/0007_alter_player_options_alter_team_options_and_more.py +40 -0
- wbcore/contrib/example_app/migrations/__init__.py +0 -0
- wbcore/contrib/example_app/models.py +978 -0
- wbcore/contrib/example_app/serializers/__init__.py +31 -0
- wbcore/contrib/example_app/serializers/league.py +112 -0
- wbcore/contrib/example_app/serializers/match_event.py +305 -0
- wbcore/contrib/example_app/serializers/person_team.py +281 -0
- wbcore/contrib/example_app/serializers/role.py +16 -0
- wbcore/contrib/example_app/serializers/season.py +54 -0
- wbcore/contrib/example_app/serializers/sport.py +38 -0
- wbcore/contrib/example_app/serializers/stadium.py +59 -0
- wbcore/contrib/example_app/serializers/teamresult.py +44 -0
- wbcore/contrib/example_app/templates/example_app/embedded_view.html +19 -0
- wbcore/contrib/example_app/tests/__init__.py +0 -0
- wbcore/contrib/example_app/tests/conftest.py +13 -0
- wbcore/contrib/example_app/tests/e2e/__init__.py +1 -0
- wbcore/contrib/example_app/tests/e2e/e2e_example_app_utility.py +52 -0
- wbcore/contrib/example_app/tests/e2e/test_league.py +70 -0
- wbcore/contrib/example_app/tests/e2e/test_person.py +68 -0
- wbcore/contrib/example_app/tests/e2e/test_teams.py +57 -0
- wbcore/contrib/example_app/tests/signals.py +7 -0
- wbcore/contrib/example_app/tests/test_displays.py +41 -0
- wbcore/contrib/example_app/tests/test_filters.py +71 -0
- wbcore/contrib/example_app/tests/test_models/test_event.py +89 -0
- wbcore/contrib/example_app/tests/test_models/test_match.py +220 -0
- wbcore/contrib/example_app/tests/test_models/test_others.py +161 -0
- wbcore/contrib/example_app/tests/test_serializers/test_league_serializer.py +34 -0
- wbcore/contrib/example_app/tests/test_serializers/test_match_serializer.py +139 -0
- wbcore/contrib/example_app/tests/test_serializers/test_role_serializer.py +13 -0
- wbcore/contrib/example_app/tests/test_serializers/test_sport_serializer.py +14 -0
- wbcore/contrib/example_app/tests/test_serializers/test_stadium_serializer.py +14 -0
- wbcore/contrib/example_app/tests/test_serializers/test_team_result_serializer.py +30 -0
- wbcore/contrib/example_app/tests/test_serializers/test_team_serializer.py +70 -0
- wbcore/contrib/example_app/tests/test_utils.py +26 -0
- wbcore/contrib/example_app/tests/test_viewsets/test_event_viewset.py +167 -0
- wbcore/contrib/example_app/tests/test_viewsets/test_league_viewset.py +84 -0
- wbcore/contrib/example_app/tests/test_viewsets/test_match_viewset.py +65 -0
- wbcore/contrib/example_app/tests/test_viewsets/test_person_viewset.py +166 -0
- wbcore/contrib/example_app/tests/test_viewsets/test_role_viewset.py +75 -0
- wbcore/contrib/example_app/tests/test_viewsets/test_sport_viewset.py +75 -0
- wbcore/contrib/example_app/tests/test_viewsets/test_stadium_viewset.py +75 -0
- wbcore/contrib/example_app/tests/test_viewsets/test_team_viewset.py +92 -0
- wbcore/contrib/example_app/tests/test_viewsets/test_teamresult_viewset.py +58 -0
- wbcore/contrib/example_app/tests/test_viewsets/test_utils_viewsets.py +124 -0
- wbcore/contrib/example_app/urls.py +82 -0
- wbcore/contrib/example_app/utils.py +21 -0
- wbcore/contrib/example_app/views.py +6 -0
- wbcore/contrib/example_app/viewsets/__init__.py +29 -0
- wbcore/contrib/example_app/viewsets/buttons/__init__.py +2 -0
- wbcore/contrib/example_app/viewsets/buttons/person.py +19 -0
- wbcore/contrib/example_app/viewsets/buttons/team.py +24 -0
- wbcore/contrib/example_app/viewsets/displays/__init__.py +23 -0
- wbcore/contrib/example_app/viewsets/displays/event.py +187 -0
- wbcore/contrib/example_app/viewsets/displays/league.py +391 -0
- wbcore/contrib/example_app/viewsets/displays/match.py +265 -0
- wbcore/contrib/example_app/viewsets/displays/person.py +252 -0
- wbcore/contrib/example_app/viewsets/displays/role.py +17 -0
- wbcore/contrib/example_app/viewsets/displays/season.py +66 -0
- wbcore/contrib/example_app/viewsets/displays/sport.py +116 -0
- wbcore/contrib/example_app/viewsets/displays/stadium.py +150 -0
- wbcore/contrib/example_app/viewsets/displays/team.py +250 -0
- wbcore/contrib/example_app/viewsets/displays/teamresult.py +31 -0
- wbcore/contrib/example_app/viewsets/endpoints/__init__.py +12 -0
- wbcore/contrib/example_app/viewsets/endpoints/endpoints.py +80 -0
- wbcore/contrib/example_app/viewsets/event.py +288 -0
- wbcore/contrib/example_app/viewsets/league.py +73 -0
- wbcore/contrib/example_app/viewsets/match.py +120 -0
- wbcore/contrib/example_app/viewsets/menu/__init__.py +1 -0
- wbcore/contrib/example_app/viewsets/menu/menus.py +103 -0
- wbcore/contrib/example_app/viewsets/menus.py +64 -0
- wbcore/contrib/example_app/viewsets/person.py +134 -0
- wbcore/contrib/example_app/viewsets/role.py +25 -0
- wbcore/contrib/example_app/viewsets/season.py +23 -0
- wbcore/contrib/example_app/viewsets/sport.py +26 -0
- wbcore/contrib/example_app/viewsets/stadium.py +31 -0
- wbcore/contrib/example_app/viewsets/team.py +69 -0
- wbcore/contrib/example_app/viewsets/teamresult.py +107 -0
- wbcore/contrib/example_app/viewsets/titles/__init__.py +9 -0
- wbcore/contrib/example_app/viewsets/titles/event.py +42 -0
- wbcore/contrib/example_app/viewsets/titles/league.py +25 -0
- wbcore/contrib/example_app/viewsets/titles/match.py +35 -0
- wbcore/contrib/example_app/viewsets/titles/person.py +36 -0
- wbcore/contrib/example_app/viewsets/titles/role.py +14 -0
- wbcore/contrib/example_app/viewsets/titles/sport.py +14 -0
- wbcore/contrib/example_app/viewsets/titles/stadium.py +14 -0
- wbcore/contrib/example_app/viewsets/titles/team.py +25 -0
- wbcore/contrib/example_app/viewsets/titles/teamresult.py +13 -0
- wbcore/contrib/geography/__init__.py +0 -0
- wbcore/contrib/geography/admin.py +30 -0
- wbcore/contrib/geography/apps.py +5 -0
- wbcore/contrib/geography/factories.py +46 -0
- wbcore/contrib/geography/fixtures/geography.json +13454 -0
- wbcore/contrib/geography/import_export/__init__.py +0 -0
- wbcore/contrib/geography/import_export/resources/__init__.py +0 -0
- wbcore/contrib/geography/import_export/resources/geography.py +11 -0
- wbcore/contrib/geography/migrations/0001_initial.py +110 -0
- wbcore/contrib/geography/migrations/0002_geography_geography_geography_tree_i739a.py +17 -0
- wbcore/contrib/geography/migrations/__init__.py +0 -0
- wbcore/contrib/geography/models.py +150 -0
- wbcore/contrib/geography/release_notes/1_0_0.md +13 -0
- wbcore/contrib/geography/release_notes/__init__.py +0 -0
- wbcore/contrib/geography/serializers.py +29 -0
- wbcore/contrib/geography/static/geography/markdown/documentation/geography.md +16 -0
- wbcore/contrib/geography/tests/__init__.py +0 -0
- wbcore/contrib/geography/tests/conftest.py +26 -0
- wbcore/contrib/geography/tests/signals.py +7 -0
- wbcore/contrib/geography/tests/test_models.py +27 -0
- wbcore/contrib/geography/tests/test_viewsets.py +100 -0
- wbcore/contrib/geography/urls.py +13 -0
- wbcore/contrib/geography/viewsets/__init__.py +1 -0
- wbcore/contrib/geography/viewsets/buttons/__init__.py +0 -0
- wbcore/contrib/geography/viewsets/display/__init__.py +1 -0
- wbcore/contrib/geography/viewsets/display/geography.py +32 -0
- wbcore/contrib/geography/viewsets/endpoints/__init__.py +0 -0
- wbcore/contrib/geography/viewsets/geography.py +58 -0
- wbcore/contrib/geography/viewsets/menu/__init__.py +1 -0
- wbcore/contrib/geography/viewsets/menu/geography.py +8 -0
- wbcore/contrib/geography/viewsets/preview/__init__.py +1 -0
- wbcore/contrib/geography/viewsets/preview/geography.py +19 -0
- wbcore/contrib/geography/viewsets/titles/__init__.py +0 -0
- wbcore/contrib/geography/viewsets/titles/geography.py +14 -0
- wbcore/contrib/gleap/__init__.py +0 -0
- wbcore/contrib/gleap/apps.py +5 -0
- wbcore/contrib/gleap/configs.py +12 -0
- wbcore/contrib/gleap/configurations.py +6 -0
- wbcore/contrib/gleap/hashes.py +13 -0
- wbcore/contrib/gleap/tests/__init__.py +0 -0
- wbcore/contrib/gleap/tests/conftest.py +1 -0
- wbcore/contrib/gleap/tests/tests.py +29 -0
- wbcore/contrib/gleap/urls.py +8 -0
- wbcore/contrib/gleap/views.py +31 -0
- wbcore/contrib/guardian/apps.py +6 -0
- wbcore/contrib/guardian/configurations.py +3 -0
- wbcore/contrib/guardian/filters.py +21 -0
- wbcore/contrib/guardian/migrations/0001_initial.py +103 -0
- wbcore/contrib/guardian/migrations/__init__.py +0 -0
- wbcore/contrib/guardian/models/__init__.py +1 -0
- wbcore/contrib/guardian/models/mixins.py +139 -0
- wbcore/contrib/guardian/models/models.py +29 -0
- wbcore/contrib/guardian/tasks.py +11 -0
- wbcore/contrib/guardian/tests/__init__.py +0 -0
- wbcore/contrib/guardian/tests/conftest.py +1 -0
- wbcore/contrib/guardian/tests/test_model_mixins.py +92 -0
- wbcore/contrib/guardian/tests/test_tasks.py +77 -0
- wbcore/contrib/guardian/tests/test_utils.py +196 -0
- wbcore/contrib/guardian/tests/test_viewsets.py +48 -0
- wbcore/contrib/guardian/urls.py +12 -0
- wbcore/contrib/guardian/utils.py +124 -0
- wbcore/contrib/guardian/viewsets/__init__.py +1 -0
- wbcore/contrib/guardian/viewsets/configs/__init__.py +4 -0
- wbcore/contrib/guardian/viewsets/configs/buttons.py +36 -0
- wbcore/contrib/guardian/viewsets/configs/displays.py +43 -0
- wbcore/contrib/guardian/viewsets/configs/endpoints.py +25 -0
- wbcore/contrib/guardian/viewsets/configs/titles.py +16 -0
- wbcore/contrib/guardian/viewsets/mixins.py +6 -0
- wbcore/contrib/guardian/viewsets/viewsets.py +141 -0
- wbcore/contrib/i18n/__init__.py +2 -0
- wbcore/contrib/i18n/buttons.py +33 -0
- wbcore/contrib/i18n/serializers/__init__.py +0 -0
- wbcore/contrib/i18n/serializers/fields.py +20 -0
- wbcore/contrib/i18n/serializers/mixins.py +13 -0
- wbcore/contrib/i18n/tests/conftest.py +11 -0
- wbcore/contrib/i18n/tests/test_viewsets.py +67 -0
- wbcore/contrib/i18n/translation.py +141 -0
- wbcore/contrib/i18n/viewsets.py +36 -0
- wbcore/contrib/icons/__init__.py +1 -0
- wbcore/contrib/icons/apps.py +5 -0
- wbcore/contrib/icons/backends/__init__.py +5 -0
- wbcore/contrib/icons/backends/default.py +375 -0
- wbcore/contrib/icons/backends/material.py +136 -0
- wbcore/contrib/icons/icons.py +167 -0
- wbcore/contrib/icons/models.py +11 -0
- wbcore/contrib/icons/serializers.py +18 -0
- wbcore/contrib/io/__init__.py +0 -0
- wbcore/contrib/io/admin.py +151 -0
- wbcore/contrib/io/apps.py +37 -0
- wbcore/contrib/io/backends/__init__.py +3 -0
- wbcore/contrib/io/backends/abstract.py +60 -0
- wbcore/contrib/io/backends/mail.py +39 -0
- wbcore/contrib/io/backends/utils.py +39 -0
- wbcore/contrib/io/configs/__init__.py +0 -0
- wbcore/contrib/io/configs/endpoints.py +13 -0
- wbcore/contrib/io/configurations/__init__.py +1 -0
- wbcore/contrib/io/configurations/base.py +7 -0
- wbcore/contrib/io/dynamic_preferences_registry.py +25 -0
- wbcore/contrib/io/enums.py +15 -0
- wbcore/contrib/io/exceptions.py +24 -0
- wbcore/contrib/io/factories.py +205 -0
- wbcore/contrib/io/fixtures/io.json +145 -0
- wbcore/contrib/io/import_export/backends/__init__.py +3 -0
- wbcore/contrib/io/import_export/backends/mail.py +51 -0
- wbcore/contrib/io/import_export/backends/sftp.py +69 -0
- wbcore/contrib/io/import_export/backends/stream.py +92 -0
- wbcore/contrib/io/import_export/parsers/__init__.py +0 -0
- wbcore/contrib/io/import_export/parsers/base_csv.py +36 -0
- wbcore/contrib/io/import_export/parsers/resources.py +50 -0
- wbcore/contrib/io/imports.py +313 -0
- wbcore/contrib/io/locale/de/LC_MESSAGES/django.po +144 -0
- wbcore/contrib/io/locale/de/LC_MESSAGES/django.po.translated +103 -0
- wbcore/contrib/io/locale/en/LC_MESSAGES/django.po +138 -0
- wbcore/contrib/io/locale/fr/LC_MESSAGES/django.po +138 -0
- wbcore/contrib/io/management/__init__.py +13 -0
- wbcore/contrib/io/migrations/0001_initial_squashed.py +319 -0
- wbcore/contrib/io/migrations/0002_importsource_creator.py +26 -0
- wbcore/contrib/io/migrations/0003_auto_20240103_1000.py +46 -0
- wbcore/contrib/io/migrations/0004_alter_importsource_status_exportsource.py +134 -0
- wbcore/contrib/io/migrations/0005_exportsource_data_alter_exportsource_query_str_and_more.py +67 -0
- wbcore/contrib/io/migrations/0006_alter_exportsource_query_params.py +20 -0
- wbcore/contrib/io/migrations/0007_alter_exportsource_query_params.py +23 -0
- wbcore/contrib/io/migrations/0008_importsource_resource_kwargs.py +18 -0
- wbcore/contrib/io/migrations/__init__.py +0 -0
- wbcore/contrib/io/mixins.py +37 -0
- wbcore/contrib/io/models.py +1037 -0
- wbcore/contrib/io/release_notes/1_0_0.md +13 -0
- wbcore/contrib/io/release_notes/__init__.py +0 -0
- wbcore/contrib/io/resources.py +176 -0
- wbcore/contrib/io/serializers.py +137 -0
- wbcore/contrib/io/signals.py +4 -0
- wbcore/contrib/io/tasks.py +27 -0
- wbcore/contrib/io/tests/__init__.py +0 -0
- wbcore/contrib/io/tests/conftest.py +42 -0
- wbcore/contrib/io/tests/test_backends.py +134 -0
- wbcore/contrib/io/tests/test_exports.py +130 -0
- wbcore/contrib/io/tests/test_imports.py +168 -0
- wbcore/contrib/io/tests/test_models.py +396 -0
- wbcore/contrib/io/tests/test_viewsets.py +271 -0
- wbcore/contrib/io/urls.py +29 -0
- wbcore/contrib/io/utils.py +44 -0
- wbcore/contrib/io/viewset_mixins.py +271 -0
- wbcore/contrib/io/viewsets.py +139 -0
- wbcore/contrib/notifications/__init__.py +0 -0
- wbcore/contrib/notifications/admin.py +63 -0
- wbcore/contrib/notifications/apps.py +43 -0
- wbcore/contrib/notifications/backends/__init__.py +0 -0
- wbcore/contrib/notifications/backends/abstract_backend.py +11 -0
- wbcore/contrib/notifications/backends/console/__init__.py +1 -0
- wbcore/contrib/notifications/backends/console/backends.py +25 -0
- wbcore/contrib/notifications/backends/firebase/__init__.py +1 -0
- wbcore/contrib/notifications/backends/firebase/backends.py +106 -0
- wbcore/contrib/notifications/configs.py +13 -0
- wbcore/contrib/notifications/configurations.py +9 -0
- wbcore/contrib/notifications/dispatch.py +90 -0
- wbcore/contrib/notifications/factories/__init__.py +0 -0
- wbcore/contrib/notifications/factories/notification_types.py +24 -0
- wbcore/contrib/notifications/factories/notifications.py +17 -0
- wbcore/contrib/notifications/factories/tokens.py +12 -0
- wbcore/contrib/notifications/locale/de/LC_MESSAGES/django.po +66 -0
- wbcore/contrib/notifications/locale/de/LC_MESSAGES/django.po.translated +63 -0
- wbcore/contrib/notifications/locale/en/LC_MESSAGES/django.po +61 -0
- wbcore/contrib/notifications/locale/fr/LC_MESSAGES/django.po +62 -0
- wbcore/contrib/notifications/migrations/0001_initial.py +116 -0
- wbcore/contrib/notifications/migrations/0002_notificationusertoken_unique_user_token_device.py +18 -0
- wbcore/contrib/notifications/migrations/0003_notificationusertoken_updated.py +17 -0
- wbcore/contrib/notifications/migrations/0004_alter_notification_body.py +17 -0
- wbcore/contrib/notifications/migrations/0005_alter_notification_endpoint.py +17 -0
- wbcore/contrib/notifications/migrations/0006_notification_created.py +25 -0
- wbcore/contrib/notifications/migrations/0007_notificationtype_resource_button_label.py +19 -0
- wbcore/contrib/notifications/migrations/0008_notificationtype_is_lock.py +18 -0
- wbcore/contrib/notifications/migrations/0009_alter_notificationtypesetting_options_and_more.py +32 -0
- wbcore/contrib/notifications/migrations/__init__.py +0 -0
- wbcore/contrib/notifications/models/__init__.py +3 -0
- wbcore/contrib/notifications/models/notification_types.py +165 -0
- wbcore/contrib/notifications/models/notifications.py +82 -0
- wbcore/contrib/notifications/models/tokens.py +46 -0
- wbcore/contrib/notifications/release_notes/1_0_0.md +13 -0
- wbcore/contrib/notifications/release_notes/__init__.py +0 -0
- wbcore/contrib/notifications/serializers/__init__.py +5 -0
- wbcore/contrib/notifications/serializers/notification_types.py +51 -0
- wbcore/contrib/notifications/serializers/notifications.py +33 -0
- wbcore/contrib/notifications/static/notifications/service-worker.js +1 -0
- wbcore/contrib/notifications/tasks.py +57 -0
- wbcore/contrib/notifications/templates/notifications/notification_template.html +43 -0
- wbcore/contrib/notifications/tests/__init__.py +0 -0
- wbcore/contrib/notifications/tests/conftest.py +47 -0
- wbcore/contrib/notifications/tests/test_backends/__init__.py +0 -0
- wbcore/contrib/notifications/tests/test_backends/test_firebase.py +79 -0
- wbcore/contrib/notifications/tests/test_configs.py +7 -0
- wbcore/contrib/notifications/tests/test_models/__init__.py +0 -0
- wbcore/contrib/notifications/tests/test_models/test_notification_types.py +85 -0
- wbcore/contrib/notifications/tests/test_models/test_notifications.py +46 -0
- wbcore/contrib/notifications/tests/test_models/test_tokens.py +30 -0
- wbcore/contrib/notifications/tests/test_serializers/__init__.py +0 -0
- wbcore/contrib/notifications/tests/test_serializers/test_notification_types.py +59 -0
- wbcore/contrib/notifications/tests/test_serializers/test_notifications.py +24 -0
- wbcore/contrib/notifications/tests/test_tasks.py +72 -0
- wbcore/contrib/notifications/tests/test_utils.py +0 -0
- wbcore/contrib/notifications/tests/test_viewsets/__init__.py +0 -0
- wbcore/contrib/notifications/tests/test_viewsets/test_notification_types.py +120 -0
- wbcore/contrib/notifications/tests/test_viewsets/test_notifications.py +143 -0
- wbcore/contrib/notifications/urls.py +27 -0
- wbcore/contrib/notifications/utils.py +27 -0
- wbcore/contrib/notifications/views.py +63 -0
- wbcore/contrib/notifications/viewsets/__init__.py +5 -0
- wbcore/contrib/notifications/viewsets/configs/notification_types.py +49 -0
- wbcore/contrib/notifications/viewsets/configs/notifications.py +91 -0
- wbcore/contrib/notifications/viewsets/menus.py +15 -0
- wbcore/contrib/notifications/viewsets/notification_types.py +51 -0
- wbcore/contrib/notifications/viewsets/notifications.py +58 -0
- wbcore/contrib/pandas/__init__.py +0 -0
- wbcore/contrib/pandas/fields.py +172 -0
- wbcore/contrib/pandas/filters.py +118 -0
- wbcore/contrib/pandas/filterset.py +28 -0
- wbcore/contrib/pandas/metadata.py +15 -0
- wbcore/contrib/pandas/tests/__init__.py +0 -0
- wbcore/contrib/pandas/tests/test_base.py +22 -0
- wbcore/contrib/pandas/tests/test_fields/__init__.py +0 -0
- wbcore/contrib/pandas/tests/test_fields/test_number_fields.py +19 -0
- wbcore/contrib/pandas/tests/test_filters/__init__.py +0 -0
- wbcore/contrib/pandas/tests/test_filters/test_pandas.py +115 -0
- wbcore/contrib/pandas/tests/test_views.py +22 -0
- wbcore/contrib/pandas/utils.py +170 -0
- wbcore/contrib/pandas/views.py +158 -0
- wbcore/contrib/tags/__init__.py +0 -0
- wbcore/contrib/tags/admin.py +17 -0
- wbcore/contrib/tags/apps.py +9 -0
- wbcore/contrib/tags/factories.py +28 -0
- wbcore/contrib/tags/filters.py +43 -0
- wbcore/contrib/tags/migrations/0001_initial.py +62 -0
- wbcore/contrib/tags/migrations/__init__.py +0 -0
- wbcore/contrib/tags/models/__init__.py +2 -0
- wbcore/contrib/tags/models/mixins.py +27 -0
- wbcore/contrib/tags/models/tags.py +107 -0
- wbcore/contrib/tags/release_notes/1_0_0.md +13 -0
- wbcore/contrib/tags/release_notes/__init__.py +0 -0
- wbcore/contrib/tags/serializers.py +90 -0
- wbcore/contrib/tags/signals.py +19 -0
- wbcore/contrib/tags/tests/__init__.py +0 -0
- wbcore/contrib/tags/tests/conftest.py +7 -0
- wbcore/contrib/tags/tests/tests.py +144 -0
- wbcore/contrib/tags/urls.py +15 -0
- wbcore/contrib/tags/viewsets/__init__.py +10 -0
- wbcore/contrib/tags/viewsets/display.py +50 -0
- wbcore/contrib/tags/viewsets/menu.py +15 -0
- wbcore/contrib/tags/viewsets/viewsets.py +58 -0
- wbcore/contrib/workflow/__init__.py +1 -0
- wbcore/contrib/workflow/admin/__init__.py +15 -0
- wbcore/contrib/workflow/admin/condition.py +9 -0
- wbcore/contrib/workflow/admin/data.py +15 -0
- wbcore/contrib/workflow/admin/display.py +9 -0
- wbcore/contrib/workflow/admin/process.py +37 -0
- wbcore/contrib/workflow/admin/step.py +92 -0
- wbcore/contrib/workflow/admin/transition.py +9 -0
- wbcore/contrib/workflow/admin/workflow.py +9 -0
- wbcore/contrib/workflow/apps.py +25 -0
- wbcore/contrib/workflow/configs.py +11 -0
- wbcore/contrib/workflow/decorators.py +25 -0
- wbcore/contrib/workflow/dispatch.py +24 -0
- wbcore/contrib/workflow/factories/__init__.py +18 -0
- wbcore/contrib/workflow/factories/condition.py +16 -0
- wbcore/contrib/workflow/factories/data.py +23 -0
- wbcore/contrib/workflow/factories/display.py +28 -0
- wbcore/contrib/workflow/factories/process.py +70 -0
- wbcore/contrib/workflow/factories/step.py +157 -0
- wbcore/contrib/workflow/factories/transition.py +27 -0
- wbcore/contrib/workflow/factories/workflow.py +20 -0
- wbcore/contrib/workflow/filters/__init__.py +24 -0
- wbcore/contrib/workflow/filters/condition.py +24 -0
- wbcore/contrib/workflow/filters/data.py +25 -0
- wbcore/contrib/workflow/filters/process.py +165 -0
- wbcore/contrib/workflow/filters/step.py +227 -0
- wbcore/contrib/workflow/filters/transition.py +42 -0
- wbcore/contrib/workflow/filters/workflow.py +44 -0
- wbcore/contrib/workflow/fixtures/workflow.json +612 -0
- wbcore/contrib/workflow/locale/de/LC_MESSAGES/django.po +1180 -0
- wbcore/contrib/workflow/locale/de/LC_MESSAGES/django.po.translated +1326 -0
- wbcore/contrib/workflow/locale/en/LC_MESSAGES/django.po +1102 -0
- wbcore/contrib/workflow/locale/fr/LC_MESSAGES/django.po +1114 -0
- wbcore/contrib/workflow/migrations/0001_initial.py +788 -0
- wbcore/contrib/workflow/migrations/0002_alter_step_step_type.py +31 -0
- wbcore/contrib/workflow/migrations/0003_alter_condition_attribute_name_and_more.py +75 -0
- wbcore/contrib/workflow/migrations/0004_alter_userstep_assignee_method.py +27 -0
- wbcore/contrib/workflow/migrations/0005_alter_userstep_assignee_method.py +17 -0
- wbcore/contrib/workflow/migrations/__init__.py +0 -0
- wbcore/contrib/workflow/models/__init__.py +19 -0
- wbcore/contrib/workflow/models/condition.py +124 -0
- wbcore/contrib/workflow/models/data.py +242 -0
- wbcore/contrib/workflow/models/display.py +34 -0
- wbcore/contrib/workflow/models/process.py +244 -0
- wbcore/contrib/workflow/models/step.py +778 -0
- wbcore/contrib/workflow/models/transition.py +71 -0
- wbcore/contrib/workflow/models/workflow.py +317 -0
- wbcore/contrib/workflow/serializers/__init__.py +37 -0
- wbcore/contrib/workflow/serializers/condition.py +65 -0
- wbcore/contrib/workflow/serializers/data.py +136 -0
- wbcore/contrib/workflow/serializers/display.py +25 -0
- wbcore/contrib/workflow/serializers/process.py +184 -0
- wbcore/contrib/workflow/serializers/signals.py +26 -0
- wbcore/contrib/workflow/serializers/step.py +365 -0
- wbcore/contrib/workflow/serializers/transition.py +81 -0
- wbcore/contrib/workflow/serializers/workflow.py +125 -0
- wbcore/contrib/workflow/sites.py +43 -0
- wbcore/contrib/workflow/static/workflow/markdown/documentation/assignedprocessstep.md +33 -0
- wbcore/contrib/workflow/static/workflow/markdown/documentation/condition.md +24 -0
- wbcore/contrib/workflow/static/workflow/markdown/documentation/decisionstep.md +30 -0
- wbcore/contrib/workflow/static/workflow/markdown/documentation/emailstep.md +45 -0
- wbcore/contrib/workflow/static/workflow/markdown/documentation/finishstep.md +33 -0
- wbcore/contrib/workflow/static/workflow/markdown/documentation/joinstep.md +33 -0
- wbcore/contrib/workflow/static/workflow/markdown/documentation/process.md +33 -0
- wbcore/contrib/workflow/static/workflow/markdown/documentation/processstep.md +51 -0
- wbcore/contrib/workflow/static/workflow/markdown/documentation/scriptstep.md +33 -0
- wbcore/contrib/workflow/static/workflow/markdown/documentation/splitstep.md +30 -0
- wbcore/contrib/workflow/static/workflow/markdown/documentation/startstep.md +27 -0
- wbcore/contrib/workflow/static/workflow/markdown/documentation/transition.md +27 -0
- wbcore/contrib/workflow/static/workflow/markdown/documentation/userstep.md +42 -0
- wbcore/contrib/workflow/static/workflow/markdown/documentation/workflow.md +32 -0
- wbcore/contrib/workflow/templates/Test_Templates.txt +25 -0
- wbcore/contrib/workflow/tests/__init__.py +0 -0
- wbcore/contrib/workflow/tests/conftest.py +281 -0
- wbcore/contrib/workflow/tests/test_configs.py +6 -0
- wbcore/contrib/workflow/tests/test_dispatch.py +94 -0
- wbcore/contrib/workflow/tests/test_displays.py +42 -0
- wbcore/contrib/workflow/tests/test_filters.py +104 -0
- wbcore/contrib/workflow/tests/test_models/step/test_decision_step.py +82 -0
- wbcore/contrib/workflow/tests/test_models/step/test_email_step.py +85 -0
- wbcore/contrib/workflow/tests/test_models/step/test_finish_step.py +167 -0
- wbcore/contrib/workflow/tests/test_models/step/test_join_step.py +117 -0
- wbcore/contrib/workflow/tests/test_models/step/test_script_step.py +24 -0
- wbcore/contrib/workflow/tests/test_models/step/test_split_step.py +49 -0
- wbcore/contrib/workflow/tests/test_models/step/test_step.py +756 -0
- wbcore/contrib/workflow/tests/test_models/step/test_user_step.py +225 -0
- wbcore/contrib/workflow/tests/test_models/test_condition.py +103 -0
- wbcore/contrib/workflow/tests/test_models/test_data.py +134 -0
- wbcore/contrib/workflow/tests/test_models/test_process.py +98 -0
- wbcore/contrib/workflow/tests/test_models/test_transition.py +128 -0
- wbcore/contrib/workflow/tests/test_models/test_workflow.py +358 -0
- wbcore/contrib/workflow/tests/test_serializers.py +200 -0
- wbcore/contrib/workflow/tests/test_viewsets.py +300 -0
- wbcore/contrib/workflow/tests/test_workflow_assignees.py +236 -0
- wbcore/contrib/workflow/urls.py +67 -0
- wbcore/contrib/workflow/utils.py +13 -0
- wbcore/contrib/workflow/viewsets/__init__.py +17 -0
- wbcore/contrib/workflow/viewsets/buttons/__init__.py +1 -0
- wbcore/contrib/workflow/viewsets/buttons/step.py +68 -0
- wbcore/contrib/workflow/viewsets/condition.py +32 -0
- wbcore/contrib/workflow/viewsets/data.py +33 -0
- wbcore/contrib/workflow/viewsets/display/__init__.py +15 -0
- wbcore/contrib/workflow/viewsets/display/condition.py +51 -0
- wbcore/contrib/workflow/viewsets/display/data.py +51 -0
- wbcore/contrib/workflow/viewsets/display/process.py +186 -0
- wbcore/contrib/workflow/viewsets/display/step.py +454 -0
- wbcore/contrib/workflow/viewsets/display/transition.py +76 -0
- wbcore/contrib/workflow/viewsets/display/workflow.py +168 -0
- wbcore/contrib/workflow/viewsets/endpoints/__init__.py +5 -0
- wbcore/contrib/workflow/viewsets/endpoints/condition.py +9 -0
- wbcore/contrib/workflow/viewsets/endpoints/data.py +9 -0
- wbcore/contrib/workflow/viewsets/endpoints/process.py +11 -0
- wbcore/contrib/workflow/viewsets/endpoints/step.py +24 -0
- wbcore/contrib/workflow/viewsets/endpoints/transition.py +19 -0
- wbcore/contrib/workflow/viewsets/menu/__init__.py +15 -0
- wbcore/contrib/workflow/viewsets/menu/condition.py +19 -0
- wbcore/contrib/workflow/viewsets/menu/data.py +19 -0
- wbcore/contrib/workflow/viewsets/menu/process.py +19 -0
- wbcore/contrib/workflow/viewsets/menu/step.py +124 -0
- wbcore/contrib/workflow/viewsets/menu/transition.py +19 -0
- wbcore/contrib/workflow/viewsets/menu/workflow.py +19 -0
- wbcore/contrib/workflow/viewsets/process.py +175 -0
- wbcore/contrib/workflow/viewsets/step.py +229 -0
- wbcore/contrib/workflow/viewsets/titles/__init__.py +16 -0
- wbcore/contrib/workflow/viewsets/titles/condition.py +14 -0
- wbcore/contrib/workflow/viewsets/titles/data.py +14 -0
- wbcore/contrib/workflow/viewsets/titles/process.py +27 -0
- wbcore/contrib/workflow/viewsets/titles/step.py +102 -0
- wbcore/contrib/workflow/viewsets/titles/transition.py +14 -0
- wbcore/contrib/workflow/viewsets/titles/workflow.py +14 -0
- wbcore/contrib/workflow/viewsets/transition.py +58 -0
- wbcore/contrib/workflow/viewsets/workflow.py +37 -0
- wbcore/contrib/workflow/workflows/__init__.py +1 -0
- wbcore/contrib/workflow/workflows/assignees.py +88 -0
- wbcore/crontab/__init__.py +0 -0
- wbcore/crontab/serializers.py +24 -0
- wbcore/crontab/viewsets.py +10 -0
- wbcore/dispatch.py +56 -0
- wbcore/docs/__init__.py +23 -0
- wbcore/docs/orderable.md +29 -0
- wbcore/docs/reparent.md +13 -0
- wbcore/dynamic_preferences_registry.py +131 -0
- wbcore/enums.py +50 -0
- wbcore/filters/__init__.py +21 -0
- wbcore/filters/backends.py +19 -0
- wbcore/filters/defaults.py +70 -0
- wbcore/filters/fields/__init__.py +15 -0
- wbcore/filters/fields/booleans.py +7 -0
- wbcore/filters/fields/choices.py +60 -0
- wbcore/filters/fields/content_type.py +45 -0
- wbcore/filters/fields/datetime.py +117 -0
- wbcore/filters/fields/models.py +123 -0
- wbcore/filters/fields/multiple_lookups.py +20 -0
- wbcore/filters/fields/numbers.py +48 -0
- wbcore/filters/fields/text.py +7 -0
- wbcore/filters/filterset.py +250 -0
- wbcore/filters/lookups.py +41 -0
- wbcore/filters/mixins.py +111 -0
- wbcore/filters/utils.py +21 -0
- wbcore/forms.py +125 -0
- wbcore/frontend.py +23 -0
- wbcore/frontend_user_configuration.py +96 -0
- wbcore/fsm/__init__.py +0 -0
- wbcore/fsm/markdown_extensions.py +31 -0
- wbcore/fsm/mixins.py +128 -0
- wbcore/locale/de/LC_MESSAGES/django.po +1252 -0
- wbcore/locale/de/LC_MESSAGES/django.po.translated +1580 -0
- wbcore/locale/en/LC_MESSAGES/django.po +1234 -0
- wbcore/locale/fr/LC_MESSAGES/django.po +1235 -0
- wbcore/management/__init__.py +88 -0
- wbcore/management/commands/__init__.py +0 -0
- wbcore/management/commands/bootstrap.py +18 -0
- wbcore/management/commands/clean_obsolete_object.py +21 -0
- wbcore/management/commands/handle_release_notes.py +54 -0
- wbcore/markdown/__init__.py +1 -0
- wbcore/markdown/admin.py +9 -0
- wbcore/markdown/dynamic_preferences_registry.py +16 -0
- wbcore/markdown/models.py +41 -0
- wbcore/markdown/template.py +38 -0
- wbcore/markdown/utils.py +37 -0
- wbcore/markdown/views.py +50 -0
- wbcore/menus/__init__.py +2 -0
- wbcore/menus/menus.py +96 -0
- wbcore/menus/registry.py +28 -0
- wbcore/menus/views.py +41 -0
- wbcore/messages.py +60 -0
- wbcore/metadata/__init__.py +0 -0
- wbcore/metadata/configs/__init__.py +0 -0
- wbcore/metadata/configs/base.py +86 -0
- wbcore/metadata/configs/buttons/__init__.py +4 -0
- wbcore/metadata/configs/buttons/bases.py +89 -0
- wbcore/metadata/configs/buttons/buttons.py +138 -0
- wbcore/metadata/configs/buttons/enums.py +68 -0
- wbcore/metadata/configs/buttons/metadata.py +8 -0
- wbcore/metadata/configs/buttons/view_config.py +145 -0
- wbcore/metadata/configs/display/__init__.py +4 -0
- wbcore/metadata/configs/display/configs.py +16 -0
- wbcore/metadata/configs/display/display.py +234 -0
- wbcore/metadata/configs/display/formatting.py +47 -0
- wbcore/metadata/configs/display/instance_display/__init__.py +15 -0
- wbcore/metadata/configs/display/instance_display/display.py +40 -0
- wbcore/metadata/configs/display/instance_display/enums.py +7 -0
- wbcore/metadata/configs/display/instance_display/layouts/__init__.py +3 -0
- wbcore/metadata/configs/display/instance_display/layouts/inlines.py +58 -0
- wbcore/metadata/configs/display/instance_display/layouts/layouts.py +77 -0
- wbcore/metadata/configs/display/instance_display/layouts/sections.py +43 -0
- wbcore/metadata/configs/display/instance_display/operators.py +22 -0
- wbcore/metadata/configs/display/instance_display/pages.py +40 -0
- wbcore/metadata/configs/display/instance_display/shortcuts.py +81 -0
- wbcore/metadata/configs/display/instance_display/signals.py +3 -0
- wbcore/metadata/configs/display/instance_display/styles.py +42 -0
- wbcore/metadata/configs/display/instance_display/utils.py +74 -0
- wbcore/metadata/configs/display/list_display.py +290 -0
- wbcore/metadata/configs/display/models.py +26 -0
- wbcore/metadata/configs/display/view_config.py +85 -0
- wbcore/metadata/configs/display/views.py +48 -0
- wbcore/metadata/configs/display/windows.py +28 -0
- wbcore/metadata/configs/documentations.py +11 -0
- wbcore/metadata/configs/endpoints.py +189 -0
- wbcore/metadata/configs/fields.py +22 -0
- wbcore/metadata/configs/filter_fields.py +41 -0
- wbcore/metadata/configs/identifiers.py +27 -0
- wbcore/metadata/configs/ordering_fields.py +24 -0
- wbcore/metadata/configs/paginations.py +11 -0
- wbcore/metadata/configs/preview.py +41 -0
- wbcore/metadata/configs/primary_keys.py +9 -0
- wbcore/metadata/configs/search_fields.py +9 -0
- wbcore/metadata/configs/titles.py +48 -0
- wbcore/metadata/configs/window_types.py +13 -0
- wbcore/metadata/exceptions.py +0 -0
- wbcore/metadata/metadata.py +34 -0
- wbcore/metadata/mixins.py +106 -0
- wbcore/metadata/tests/__init__.py +0 -0
- wbcore/metadata/tests/test_buttons.py +165 -0
- wbcore/metadata/utils.py +4 -0
- wbcore/migrations/0001_initial_squashed_squashed_0010_preset_appliedpreset.py +398 -0
- wbcore/migrations/0011_genericmodel.py +22 -0
- wbcore/migrations/0012_delete_notification.py +15 -0
- wbcore/migrations/0013_delete_colorgradient.py +14 -0
- wbcore/migrations/0014_biguserobjectpermission_system.py +44 -0
- wbcore/migrations/__init__.py +0 -0
- wbcore/models/__init__.py +6 -0
- wbcore/models/base.py +184 -0
- wbcore/models/fields.py +29 -0
- wbcore/models/orderable.py +6 -0
- wbcore/pagination.py +64 -0
- wbcore/pandas/__init__.py +53 -0
- wbcore/pandas/fields.py +25 -0
- wbcore/pandas/filterset.py +9 -0
- wbcore/pandas/utils.py +25 -0
- wbcore/pandas/views.py +9 -0
- wbcore/permissions/__init__.py +0 -0
- wbcore/permissions/backend.py +36 -0
- wbcore/permissions/mixins.py +72 -0
- wbcore/permissions/permissions.py +51 -0
- wbcore/permissions/registry.py +33 -0
- wbcore/permissions/shortcuts.py +38 -0
- wbcore/permissions/utils.py +26 -0
- wbcore/release_notes/__init__.py +0 -0
- wbcore/release_notes/admin.py +30 -0
- wbcore/release_notes/buttons.py +26 -0
- wbcore/release_notes/display.py +52 -0
- wbcore/release_notes/filters.py +36 -0
- wbcore/release_notes/models.py +66 -0
- wbcore/release_notes/serializers.py +12 -0
- wbcore/release_notes/utils.py +14 -0
- wbcore/release_notes/viewsets.py +52 -0
- wbcore/reversion/__init__.py +0 -0
- wbcore/reversion/filters.py +39 -0
- wbcore/reversion/serializers.py +81 -0
- wbcore/reversion/viewsets/__init__.py +6 -0
- wbcore/reversion/viewsets/buttons.py +95 -0
- wbcore/reversion/viewsets/displays.py +45 -0
- wbcore/reversion/viewsets/endpoints.py +30 -0
- wbcore/reversion/viewsets/titles.py +19 -0
- wbcore/reversion/viewsets/viewsets.py +112 -0
- wbcore/routers.py +63 -0
- wbcore/search/__init__.py +62 -0
- wbcore/serializers/__init__.py +68 -0
- wbcore/serializers/fields/__init__.py +59 -0
- wbcore/serializers/fields/boolean.py +51 -0
- wbcore/serializers/fields/choice.py +60 -0
- wbcore/serializers/fields/datetime.py +178 -0
- wbcore/serializers/fields/fields.py +193 -0
- wbcore/serializers/fields/file.py +20 -0
- wbcore/serializers/fields/fsm.py +20 -0
- wbcore/serializers/fields/json.py +56 -0
- wbcore/serializers/fields/list.py +104 -0
- wbcore/serializers/fields/mixins.py +196 -0
- wbcore/serializers/fields/number.py +94 -0
- wbcore/serializers/fields/other.py +34 -0
- wbcore/serializers/fields/primary_key.py +22 -0
- wbcore/serializers/fields/related.py +150 -0
- wbcore/serializers/fields/text.py +141 -0
- wbcore/serializers/fields/types.py +46 -0
- wbcore/serializers/mixins.py +24 -0
- wbcore/serializers/serializers.py +425 -0
- wbcore/serializers/utils.py +143 -0
- wbcore/shares/__init__.py +1 -0
- wbcore/shares/config.py +64 -0
- wbcore/shares/decorator.py +13 -0
- wbcore/shares/signals.py +3 -0
- wbcore/shares/sites.py +29 -0
- wbcore/shares/views.py +25 -0
- wbcore/signals/__init__.py +7 -0
- wbcore/signals/clone.py +4 -0
- wbcore/signals/filters.py +3 -0
- wbcore/signals/instance_buttons.py +5 -0
- wbcore/signals/merge.py +4 -0
- wbcore/signals/models.py +4 -0
- wbcore/signals/permissions.py +3 -0
- wbcore/signals/serializers.py +5 -0
- wbcore/tasks.py +87 -0
- wbcore/template.py +29 -0
- wbcore/templates/errors/404.html +134 -0
- wbcore/templates/errors/500.html +138 -0
- wbcore/templates/errors/503.html +132 -0
- wbcore/templates/errors/custom.html +132 -0
- wbcore/templates/forms.py +0 -0
- wbcore/templates/notifications/email_template.html +43 -0
- wbcore/templates/reversion/__init__.py +0 -0
- wbcore/templates/reversion/compare_detail.html +19 -0
- wbcore/templates/wbcore/admin/change_list.html +8 -0
- wbcore/templates/wbcore/admin/csv_form.html +15 -0
- wbcore/templates/wbcore/dynamic_color_array.html +29 -0
- wbcore/templates/wbcore/email_base_template.html +335 -0
- wbcore/templates/wbcore/email_notification_template.html +10 -0
- wbcore/templates/wbcore/frontend.html +51 -0
- wbcore/test/__init__.py +30 -0
- wbcore/test/e2e_helpers_methods/e2e_checks.py +127 -0
- wbcore/test/e2e_helpers_methods/e2e_helper_methods.py +397 -0
- wbcore/test/mixins.py +660 -0
- wbcore/test/signals.py +6 -0
- wbcore/test/tests.py +129 -0
- wbcore/test/utils.py +227 -0
- wbcore/tests/__init__.py +0 -0
- wbcore/tests/conftest.py +55 -0
- wbcore/tests/e2e/__init__.py +0 -0
- wbcore/tests/e2e/test_e2e.py +25 -0
- wbcore/tests/models.py +6 -0
- wbcore/tests/test_cache/__init__.py +0 -0
- wbcore/tests/test_cache/test_decorators.py +28 -0
- wbcore/tests/test_cache/test_mixins.py +30 -0
- wbcore/tests/test_cache/test_registry.py +58 -0
- wbcore/tests/test_configs.py +58 -0
- wbcore/tests/test_enums.py +56 -0
- wbcore/tests/test_fields/__init__.py +0 -0
- wbcore/tests/test_fields/test_boolean_fields.py +49 -0
- wbcore/tests/test_fields/test_choice_fields.py +57 -0
- wbcore/tests/test_fields/test_datetime_fields.py +152 -0
- wbcore/tests/test_fields/test_fields.py +24 -0
- wbcore/tests/test_fields/test_file_fields.py +52 -0
- wbcore/tests/test_fields/test_json_fields.py +28 -0
- wbcore/tests/test_fields/test_list_fields.py +27 -0
- wbcore/tests/test_fields/test_mixins.py +112 -0
- wbcore/tests/test_fields/test_number_fields.py +166 -0
- wbcore/tests/test_fields/test_other_fields.py +60 -0
- wbcore/tests/test_fields/test_primary_key_fields.py +52 -0
- wbcore/tests/test_fields/test_related.py +81 -0
- wbcore/tests/test_fields/test_text_fields.py +118 -0
- wbcore/tests/test_filters/__init__.py +0 -0
- wbcore/tests/test_filters/test_mixins.py +109 -0
- wbcore/tests/test_list_display.py +29 -0
- wbcore/tests/test_models/__init__.py +0 -0
- wbcore/tests/test_models/test_fields.py +26 -0
- wbcore/tests/test_models/test_mixins.py +32 -0
- wbcore/tests/test_new_display/__init__.py +0 -0
- wbcore/tests/test_new_display/test_inlines.py +13 -0
- wbcore/tests/test_new_display/test_layouts.py +15 -0
- wbcore/tests/test_new_display/test_operators.py +16 -0
- wbcore/tests/test_new_display/test_pages.py +8 -0
- wbcore/tests/test_new_display/test_sections.py +0 -0
- wbcore/tests/test_new_display/test_shortcuts.py +38 -0
- wbcore/tests/test_new_display/test_utils.py +49 -0
- wbcore/tests/test_pagination.py +32 -0
- wbcore/tests/test_permissions/test_backend.py +29 -0
- wbcore/tests/test_serializers/__init__.py +0 -0
- wbcore/tests/test_serializers/test_fields.py +141 -0
- wbcore/tests/test_serializers/test_mixins.py +54 -0
- wbcore/tests/test_serializers/test_related.py +78 -0
- wbcore/tests/test_something.py +39 -0
- wbcore/tests/test_utils/__init__.py +0 -0
- wbcore/tests/test_utils/test_date.py +50 -0
- wbcore/tests/test_utils/test_date_builder.py +124 -0
- wbcore/tests/test_utils/test_primary.py +80 -0
- wbcore/tests/test_utils/test_signals.py +40 -0
- wbcore/tests/test_viewsets.py +21 -0
- wbcore/urls.py +117 -0
- wbcore/utils/__init__.py +13 -0
- wbcore/utils/cache.py +8 -0
- wbcore/utils/date.py +237 -0
- wbcore/utils/date_builder/__init__.py +18 -0
- wbcore/utils/date_builder/components.py +42 -0
- wbcore/utils/date_builder/offsets.py +27 -0
- wbcore/utils/deprecations.py +11 -0
- wbcore/utils/enum.py +23 -0
- wbcore/utils/figures.py +286 -0
- wbcore/utils/html.py +8 -0
- wbcore/utils/importlib.py +13 -0
- wbcore/utils/itertools.py +40 -0
- wbcore/utils/models.py +293 -0
- wbcore/utils/numbers.py +69 -0
- wbcore/utils/prettytable.py +35 -0
- wbcore/utils/print.py +29 -0
- wbcore/utils/renderers.py +13 -0
- wbcore/utils/reportlab.py +7 -0
- wbcore/utils/rrules.py +82 -0
- wbcore/utils/serializers.py +9 -0
- wbcore/utils/settings.py +5 -0
- wbcore/utils/signals.py +36 -0
- wbcore/utils/string_loader.py +42 -0
- wbcore/utils/strings.py +77 -0
- wbcore/utils/urls.py +70 -0
- wbcore/utils/views.py +207 -0
- wbcore/views.py +26 -0
- wbcore/viewsets/__init__.py +12 -0
- wbcore/viewsets/encoders.py +18 -0
- wbcore/viewsets/generics.py +5 -0
- wbcore/viewsets/mixins.py +295 -0
- wbcore/viewsets/utils.py +27 -0
- wbcore/viewsets/viewsets.py +202 -0
- wbcore/workers.py +8 -0
- wbcore-1.59.9.dist-info/METADATA +65 -0
- wbcore-1.59.9.dist-info/RECORD +1239 -0
- wbcore-1.59.9.dist-info/WHEEL +5 -0
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
import urllib
|
|
2
|
+
from contextlib import suppress
|
|
3
|
+
|
|
4
|
+
from django.contrib.auth import get_user_model
|
|
5
|
+
from django.core.validators import URLValidator, ValidationError
|
|
6
|
+
from django.db import models
|
|
7
|
+
from django.urls import Resolver404, resolve
|
|
8
|
+
from django.utils.functional import cached_property
|
|
9
|
+
|
|
10
|
+
from wbcore.contrib.notifications.utils import base_domain, create_notification_type
|
|
11
|
+
|
|
12
|
+
|
|
13
|
+
class Notification(models.Model):
|
|
14
|
+
title = models.CharField(max_length=255)
|
|
15
|
+
body = models.TextField(null=True, blank=True)
|
|
16
|
+
|
|
17
|
+
user = models.ForeignKey(to=get_user_model(), related_name="notifications_notifications", on_delete=models.CASCADE)
|
|
18
|
+
notification_type = models.ForeignKey(
|
|
19
|
+
to="notifications.NotificationType", related_name="notifications", on_delete=models.CASCADE
|
|
20
|
+
)
|
|
21
|
+
endpoint = models.CharField(max_length=2048, null=True, blank=True)
|
|
22
|
+
|
|
23
|
+
created = models.DateTimeField(auto_now_add=True)
|
|
24
|
+
sent = models.DateTimeField(null=True, blank=True)
|
|
25
|
+
read = models.DateTimeField(null=True, blank=True)
|
|
26
|
+
|
|
27
|
+
def __str__(self) -> str:
|
|
28
|
+
return f"{self.user} {self.title}"
|
|
29
|
+
|
|
30
|
+
class Meta:
|
|
31
|
+
verbose_name = "Notification"
|
|
32
|
+
verbose_name_plural = "Notifications"
|
|
33
|
+
|
|
34
|
+
notification_types = [
|
|
35
|
+
create_notification_type(
|
|
36
|
+
"workbench.system",
|
|
37
|
+
"System Notifications",
|
|
38
|
+
"System Notifications.",
|
|
39
|
+
True,
|
|
40
|
+
True,
|
|
41
|
+
False,
|
|
42
|
+
),
|
|
43
|
+
]
|
|
44
|
+
|
|
45
|
+
@cached_property
|
|
46
|
+
def is_endpoint_internal(self) -> bool:
|
|
47
|
+
with suppress(Resolver404):
|
|
48
|
+
if self.endpoint:
|
|
49
|
+
resolve(
|
|
50
|
+
urllib.parse.urlsplit(urllib.parse.unquote(self.endpoint)).path
|
|
51
|
+
) # we need to truncate query parameters
|
|
52
|
+
return True
|
|
53
|
+
return False
|
|
54
|
+
|
|
55
|
+
@cached_property
|
|
56
|
+
def is_endpoint_valid(self) -> bool:
|
|
57
|
+
try:
|
|
58
|
+
URLValidator()(self.endpoint)
|
|
59
|
+
return True
|
|
60
|
+
except ValidationError:
|
|
61
|
+
return False
|
|
62
|
+
|
|
63
|
+
def get_full_endpoint(self, as_shareable_internal_link: bool = False) -> str | None:
|
|
64
|
+
if self.is_endpoint_internal:
|
|
65
|
+
if as_shareable_internal_link:
|
|
66
|
+
return f"{base_domain()}?widget_endpoint={self.endpoint}"
|
|
67
|
+
else:
|
|
68
|
+
return f"{base_domain()}{self.endpoint}"
|
|
69
|
+
elif self.is_endpoint_valid:
|
|
70
|
+
return self.endpoint
|
|
71
|
+
|
|
72
|
+
@classmethod
|
|
73
|
+
def get_endpoint_basename(cls) -> str:
|
|
74
|
+
return "wbcore:notifications:notification"
|
|
75
|
+
|
|
76
|
+
@classmethod
|
|
77
|
+
def get_representation_value_key(cls) -> str:
|
|
78
|
+
return "id"
|
|
79
|
+
|
|
80
|
+
@classmethod
|
|
81
|
+
def get_representation_label_key(cls) -> str:
|
|
82
|
+
return "{{title}}"
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
from django.contrib.auth import get_user_model
|
|
2
|
+
from django.db import models
|
|
3
|
+
from django.db.models.query import QuerySet
|
|
4
|
+
|
|
5
|
+
from wbcore.contrib.notifications.models.notification_types import (
|
|
6
|
+
NotificationTypeSetting,
|
|
7
|
+
)
|
|
8
|
+
|
|
9
|
+
|
|
10
|
+
class NotificationUserTokenManager(models.Manager):
|
|
11
|
+
def filter_for_user_settings(self, user_setting: NotificationTypeSetting) -> QuerySet["NotificationUserToken"]:
|
|
12
|
+
device_type = NotificationUserToken.NotificationDeviceType
|
|
13
|
+
device_types = []
|
|
14
|
+
|
|
15
|
+
if user_setting.enable_web:
|
|
16
|
+
device_types.append(device_type.WEB)
|
|
17
|
+
|
|
18
|
+
if user_setting.enable_mobile:
|
|
19
|
+
device_types.append(device_type.MOBILE)
|
|
20
|
+
|
|
21
|
+
return self.filter(
|
|
22
|
+
user=user_setting.user,
|
|
23
|
+
device_type__in=device_types,
|
|
24
|
+
)
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
class NotificationUserToken(models.Model):
|
|
28
|
+
class NotificationDeviceType(models.TextChoices):
|
|
29
|
+
WEB = "WEB", "Web"
|
|
30
|
+
MOBILE = "MOBILE", "Mobile"
|
|
31
|
+
|
|
32
|
+
user = models.ForeignKey(to=get_user_model(), related_name="notifications_tokens", on_delete=models.CASCADE)
|
|
33
|
+
token = models.CharField(max_length=256)
|
|
34
|
+
device_type = models.CharField(max_length=16, choices=NotificationDeviceType.choices)
|
|
35
|
+
|
|
36
|
+
updated = models.DateTimeField(auto_now=True)
|
|
37
|
+
|
|
38
|
+
objects = NotificationUserTokenManager()
|
|
39
|
+
|
|
40
|
+
def __str__(self) -> str:
|
|
41
|
+
return f"{self.user}: {self.token} ({self.device_type})"
|
|
42
|
+
|
|
43
|
+
class Meta:
|
|
44
|
+
constraints = [
|
|
45
|
+
models.UniqueConstraint(fields=["user", "token", "device_type"], name="unique_user_token_device")
|
|
46
|
+
]
|
|
File without changes
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
from rest_framework.reverse import reverse
|
|
2
|
+
|
|
3
|
+
from wbcore import serializers
|
|
4
|
+
from wbcore.contrib.notifications.models import (
|
|
5
|
+
NotificationType,
|
|
6
|
+
NotificationTypeSetting,
|
|
7
|
+
)
|
|
8
|
+
|
|
9
|
+
|
|
10
|
+
class NotificationTypeRepresentationSerializer(serializers.RepresentationSerializer):
|
|
11
|
+
class Meta:
|
|
12
|
+
model = NotificationType
|
|
13
|
+
fields = (
|
|
14
|
+
"id",
|
|
15
|
+
"code",
|
|
16
|
+
"title",
|
|
17
|
+
"help_text",
|
|
18
|
+
)
|
|
19
|
+
|
|
20
|
+
|
|
21
|
+
class NotificationTypeSettingModelSerializer(serializers.ModelSerializer):
|
|
22
|
+
_notification_type = NotificationTypeRepresentationSerializer(source="notification_type")
|
|
23
|
+
help_text = serializers.CharField()
|
|
24
|
+
locked_icon = serializers.IconSelectField(read_only=True)
|
|
25
|
+
locked = serializers.BooleanField(read_only=True)
|
|
26
|
+
_update_url = serializers.SerializerMethodField()
|
|
27
|
+
|
|
28
|
+
def get__update_url(self, obj):
|
|
29
|
+
if not obj.notification_type.is_lock:
|
|
30
|
+
return reverse(
|
|
31
|
+
"wbcore:notifications:notification_type_setting-list", args=[], request=self.context["request"]
|
|
32
|
+
)
|
|
33
|
+
return None
|
|
34
|
+
|
|
35
|
+
class Meta:
|
|
36
|
+
read_only_fields = ("user", "notification_type", "help_text")
|
|
37
|
+
model = NotificationTypeSetting
|
|
38
|
+
fields = (
|
|
39
|
+
"id",
|
|
40
|
+
"notification_type",
|
|
41
|
+
"_notification_type",
|
|
42
|
+
"help_text",
|
|
43
|
+
"user",
|
|
44
|
+
"enable_web",
|
|
45
|
+
"enable_mobile",
|
|
46
|
+
"enable_email",
|
|
47
|
+
"locked",
|
|
48
|
+
"locked_icon",
|
|
49
|
+
"_additional_resources",
|
|
50
|
+
"_update_url",
|
|
51
|
+
)
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
from wbcore import serializers
|
|
2
|
+
from wbcore.contrib.notifications.models import Notification
|
|
3
|
+
from wbcore.contrib.notifications.serializers.notification_types import (
|
|
4
|
+
NotificationTypeRepresentationSerializer,
|
|
5
|
+
)
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
class NotificationModelSerializer(serializers.ModelSerializer):
|
|
9
|
+
_notification_type = NotificationTypeRepresentationSerializer(source="notification_type")
|
|
10
|
+
|
|
11
|
+
@serializers.register_resource()
|
|
12
|
+
def register_buttons(self, instance, request, user) -> dict[str, str]:
|
|
13
|
+
if endpoint := instance.endpoint:
|
|
14
|
+
if instance.is_endpoint_internal:
|
|
15
|
+
return {"open_internal_resource": endpoint}
|
|
16
|
+
else:
|
|
17
|
+
return {"open_external_resource": endpoint}
|
|
18
|
+
return {}
|
|
19
|
+
|
|
20
|
+
class Meta:
|
|
21
|
+
model = Notification
|
|
22
|
+
fields = read_only_fields = (
|
|
23
|
+
"id",
|
|
24
|
+
"notification_type",
|
|
25
|
+
"_notification_type",
|
|
26
|
+
"title",
|
|
27
|
+
"body",
|
|
28
|
+
"user",
|
|
29
|
+
"endpoint",
|
|
30
|
+
"sent",
|
|
31
|
+
"read",
|
|
32
|
+
"_additional_resources",
|
|
33
|
+
)
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
function a0_0x4393(){const _0x57af5e=['816WHdKTb','body','initializeApp','[101,121,74,104,99,71,108,76,90,88,107,105,79,105,74,66,83,88,112,104,85,51,108,69,98,86,100,52,89,88,100,119,78,69,104,120,86,110,108,52,99,48,104,52,85,108,70,88,77,87,119,53,86,51,70,77,85,110,74,112,99,70,90,89,77,71,115,105,76,67,74,104,100,88,82,111,82,71,57,116,89,87,108,117,73,106,111,105,100,50,57,121,97,50,74,108,98,109,78,111,76,88,78,48,89,87,108,117,98,72,107,117,90,109,108,121,90,87,74,104,99,50,86,104,99,72,65,117,89,50,57,116,73,105,119,105,99,72,74,118,97,109,86,106,100,69,108,107,73,106,111,105,100,50,57,121,97,50,74,108,98,109,78,111,76,88,78,48,89,87,108,117,98,72,107,105,76,67,74,122,100,71,57,121,89,87,100,108,81,110,86,106,97,50,86,48,73,106,111,105,100,50,57,121,97,50,74,108,98,109,78,111,76,88,78,48,89,87,108,117,98,72,107,117,89,88,66,119,99,51,66,118,100,67,53,106,98,50,48,105,76,67,74,116,90,88,78,122,89,87,100,112,98,109,100,84,90,87,53,107,90,88,74,74,90,67,73,54,73,106,81,49,78,106,69,53,79,68,103,49,77,68,77,49,77,105,73,115,73,109,70,119,99,69,108,107,73,106,111,105,77,84,111,48,78,84,89,120,79,84,103,52,78,84,65,122,78,84,73,54,100,50,86,105,79,109,90,104,89,87,73,120,90,68,82,106,90,68,78,109,89,106,100,109,78,122,65,51,89,84,90,104,90,106,85,105,76,67,74,116,90,87,70,122,100,88,74,108,98,87,86,117,100,69,108,107,73,106,111,105,82,121,49,88,78,107,52,51,78,106,89,51,78,69,82,76,73,110,48,61]','find','focus','origin','notificationclick','https://www.gstatic.com/firebasejs/9.22.0/firebase-messaging-compat.js','/?widget_endpoint=','waitUntil','title','window','openWindow','2928488LdzTTp','false','https://www.gstatic.com/firebasejs/9.22.0/firebase-app-compat.js','notification','is_endpoint_internal','close','9347456hpzmwS','url','HANDLE_NOTIFICATION','2095ejcNAP','3666918SdTsTm','497VBfvBu','parse','onBackgroundMessage','2684wPivrd','showNotification','data','location','3973545Klehgi','decode','endpoint','542235lhCfqf'];a0_0x4393=function(){return _0x57af5e;};return a0_0x4393();}function a0_0x3412(_0x3e53c1,_0x48472e){const _0x439370=a0_0x4393();return a0_0x3412=function(_0x3412c6,_0x54b59f){_0x3412c6=_0x3412c6-0x156;let _0x3fb5d8=_0x439370[_0x3412c6];return _0x3fb5d8;},a0_0x3412(_0x3e53c1,_0x48472e);}(function(_0x35e4fd,_0x10da15){const _0x7c8df9=a0_0x3412,_0x3a8c24=_0x35e4fd();while(!![]){try{const _0xb925b3=parseInt(_0x7c8df9(0x164))/0x1+-parseInt(_0x7c8df9(0x173))/0x2+parseInt(_0x7c8df9(0x159))/0x3+-parseInt(_0x7c8df9(0x15d))/0x4*(parseInt(_0x7c8df9(0x158))/0x5)+parseInt(_0x7c8df9(0x165))/0x6*(parseInt(_0x7c8df9(0x15a))/0x7)+parseInt(_0x7c8df9(0x179))/0x8+-parseInt(_0x7c8df9(0x161))/0x9;if(_0xb925b3===_0x10da15)break;else _0x3a8c24['push'](_0x3a8c24['shift']());}catch(_0x23d016){_0x3a8c24['push'](_0x3a8c24['shift']());}}}(a0_0x4393,0xb8813),((()=>{const _0x3232c0=a0_0x3412;var _0x36e6f1={};importScripts(_0x3232c0(0x175)),importScripts(_0x3232c0(0x16d));const _0x100673=_0x3232c0(0x168),_0x5107be=new Uint8Array(JSON[_0x3232c0(0x15b)](_0x100673)),_0x56e78c=new TextDecoder(),_0x63139d=_0x56e78c[_0x3232c0(0x162)](_0x5107be),_0x467794=JSON[_0x3232c0(0x15b)](atob(_0x63139d)),_0x3d92ff=firebase[_0x3232c0(0x167)](_0x467794),_0x21f609=firebase['messaging'](_0x3d92ff);_0x21f609[_0x3232c0(0x15c)](_0x2180f1=>{const _0x366f43=_0x3232c0;var _0x4972d6,_0xfb3e9b;const {data:_0x2e4b1c}=_0x2180f1,_0x2aae25=(_0x4972d6=_0x2e4b1c==null?void 0x0:_0x2e4b1c[_0x366f43(0x170)])!=null?_0x4972d6:'',_0x277939={'body':(_0xfb3e9b=_0x2e4b1c==null?void 0x0:_0x2e4b1c[_0x366f43(0x166)])!=null?_0xfb3e9b:'','icon':'https://stainly-cdn.fra1.cdn.digitaloceanspaces.com/media%2Flogo_orange.svg','data':{'endpoint':_0x2e4b1c==null?void 0x0:_0x2e4b1c[_0x366f43(0x163)],'is_endpoint_internal':_0x2e4b1c==null?void 0x0:_0x2e4b1c[_0x366f43(0x177)]}};return self['registration'][_0x366f43(0x15e)](_0x2aae25,_0x277939);}),self['addEventListener'](_0x3232c0(0x16c),_0x541ced=>{const _0x2767d9=_0x3232c0;var _0x47b9e3,_0x21a538;_0x541ced['notification'][_0x2767d9(0x178)]();const _0x4b09fb=((_0x47b9e3=_0x541ced[_0x2767d9(0x176)]['data'])==null?void 0x0:_0x47b9e3['endpoint'])||'',_0x48f331=(_0x21a538=_0x541ced[_0x2767d9(0x176)][_0x2767d9(0x15f)])==null?void 0x0:_0x21a538[_0x2767d9(0x177)],{origin:_0xda61ed}=self[_0x2767d9(0x160)];_0x541ced[_0x2767d9(0x16f)](((async()=>{const _0x30d532=_0x2767d9,_0x2cc1d4=await clients['matchAll']({'type':_0x30d532(0x171),'includeUncontrolled':!![]}),_0x16e94d=_0x2cc1d4[_0x30d532(0x169)](_0x44d3d3=>new URL(_0x44d3d3[_0x30d532(0x156)])[_0x30d532(0x16b)]===_0xda61ed&&_0x30d532(0x16a)in _0x44d3d3);if(_0x16e94d)return _0x16e94d['postMessage']({'type':_0x30d532(0x157),'endpoint':_0x4b09fb,'is_endpoint_internal':_0x48f331}),_0x16e94d[_0x30d532(0x16a)]();else{if(clients[_0x30d532(0x172)])return clients[_0x30d532(0x172)](_0x48f331===_0x30d532(0x174)?_0x4b09fb:new URL(_0x4b09fb)[_0x30d532(0x16b)]+_0x30d532(0x16e)+encodeURIComponent(_0x4b09fb));}})()));});})()));
|
|
@@ -0,0 +1,57 @@
|
|
|
1
|
+
from celery import shared_task
|
|
2
|
+
from django.conf import settings
|
|
3
|
+
from django.core.mail.message import EmailMultiAlternatives
|
|
4
|
+
from django.template.loader import get_template
|
|
5
|
+
from django.utils.html import strip_tags
|
|
6
|
+
from django.utils.module_loading import import_string
|
|
7
|
+
|
|
8
|
+
from wbcore.contrib.notifications.models.notification_types import (
|
|
9
|
+
NotificationTypeSetting,
|
|
10
|
+
)
|
|
11
|
+
from wbcore.contrib.notifications.models.notifications import Notification
|
|
12
|
+
from wbcore.workers import Queue
|
|
13
|
+
|
|
14
|
+
|
|
15
|
+
def send_notification_email(notification: Notification):
|
|
16
|
+
"""Sends out a notification to the user specified inside the notification
|
|
17
|
+
|
|
18
|
+
Args:
|
|
19
|
+
notification: The notification that is going to be send
|
|
20
|
+
"""
|
|
21
|
+
|
|
22
|
+
context = {
|
|
23
|
+
"title": notification.title,
|
|
24
|
+
"message": notification.body or "",
|
|
25
|
+
"notification_share_url": notification.get_full_endpoint(as_shareable_internal_link=True),
|
|
26
|
+
"notification_endpoint": notification.get_full_endpoint(),
|
|
27
|
+
}
|
|
28
|
+
rendered_template = get_template("notifications/notification_template.html").render(context)
|
|
29
|
+
msg = EmailMultiAlternatives(
|
|
30
|
+
subject=notification.title,
|
|
31
|
+
body=strip_tags(rendered_template),
|
|
32
|
+
from_email=getattr(settings, "WBCORE_NOTIFICATION_EMAIL_FROM", "no_reply@stainly.com"),
|
|
33
|
+
to=[notification.user.email], # type: ignore
|
|
34
|
+
)
|
|
35
|
+
msg.attach_alternative(rendered_template, "text/html")
|
|
36
|
+
msg.send()
|
|
37
|
+
|
|
38
|
+
|
|
39
|
+
@shared_task(queue=Queue.HIGH_PRIORITY.value)
|
|
40
|
+
def send_notification_task(notification_pk: int):
|
|
41
|
+
"""A celery task to send out a notification via email, web or mobile
|
|
42
|
+
|
|
43
|
+
Args:
|
|
44
|
+
notification_pk: The primary key of the notification that is going to be send out
|
|
45
|
+
"""
|
|
46
|
+
|
|
47
|
+
notification = Notification.objects.get(pk=notification_pk)
|
|
48
|
+
|
|
49
|
+
notification_user_settings = NotificationTypeSetting.objects.get(
|
|
50
|
+
notification_type=notification.notification_type,
|
|
51
|
+
user=notification.user,
|
|
52
|
+
)
|
|
53
|
+
if notification_user_settings.enable_email:
|
|
54
|
+
send_notification_email(notification)
|
|
55
|
+
|
|
56
|
+
backend = import_string(settings.NOTIFICATION_BACKEND)
|
|
57
|
+
backend.send_notification(notification)
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
{% extends "wbcore/email_base_template.html" %}
|
|
2
|
+
{% load i18n %}
|
|
3
|
+
{% block body %}
|
|
4
|
+
<tr class="content-row">
|
|
5
|
+
<td style="background-color: #eaf2ff;"></td>
|
|
6
|
+
<td class="body-content">
|
|
7
|
+
<p>
|
|
8
|
+
<b>{{ title }}</b>
|
|
9
|
+
</p>
|
|
10
|
+
</td>
|
|
11
|
+
<td style="background-color: #eaf2ff;"></td>
|
|
12
|
+
</tr>
|
|
13
|
+
<tr class="content-row">
|
|
14
|
+
<td style="background-color: #eaf2ff;"></td>
|
|
15
|
+
<td class="body-content">
|
|
16
|
+
{{ message | safe}}
|
|
17
|
+
</td>
|
|
18
|
+
<td style="background-color: #eaf2ff;"></td>
|
|
19
|
+
</tr>
|
|
20
|
+
{% if notification_share_url %}
|
|
21
|
+
<tr class="content-row">
|
|
22
|
+
<td style="background-color: #eaf2ff;"></td>
|
|
23
|
+
<td class="body-content">
|
|
24
|
+
<table cellspacing="0" cellpadding="0">
|
|
25
|
+
<tr>
|
|
26
|
+
<td style="border-radius: 10px;" bgcolor="#f76f5b">
|
|
27
|
+
<a class="link" href="{{ notification_share_url }}" target="_blank" style="padding: 8px 12px; border: 1px solid #ED2939; border-radius: 10px;font-family: Helvetica, Arial, sans-serif;font-size: 14px; color: #ffffff;text-decoration: none;font-weight:bold;display: inline-block;">
|
|
28
|
+
{% translate "Open" %}
|
|
29
|
+
</a>
|
|
30
|
+
</td>
|
|
31
|
+
</tr>
|
|
32
|
+
</table>
|
|
33
|
+
</td>
|
|
34
|
+
<td style="background-color: #eaf2ff;"></td>
|
|
35
|
+
</tr>
|
|
36
|
+
<tr>
|
|
37
|
+
<td style="background-color: #eaf2ff;"></td>
|
|
38
|
+
<td class="body-content">
|
|
39
|
+
</td>
|
|
40
|
+
<td style="background-color: #eaf2ff;"></td>
|
|
41
|
+
</tr>
|
|
42
|
+
{% endif %}
|
|
43
|
+
{% endblock %}
|
|
File without changes
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import pytest
|
|
2
|
+
from pytest_factoryboy import register
|
|
3
|
+
from rest_framework.test import APIClient, APIRequestFactory
|
|
4
|
+
from wbcore.contrib.authentication.factories import UserFactory
|
|
5
|
+
from wbcore.contrib.directory.factories import PersonFactory
|
|
6
|
+
from wbcore.contrib.notifications.factories.notification_types import (
|
|
7
|
+
NotificationTypeModelFactory,
|
|
8
|
+
NotificationTypeSettingModelFactory,
|
|
9
|
+
)
|
|
10
|
+
from wbcore.contrib.notifications.factories.notifications import (
|
|
11
|
+
NotificationModelFactory,
|
|
12
|
+
)
|
|
13
|
+
from wbcore.contrib.notifications.factories.tokens import (
|
|
14
|
+
NotificationUserTokenModelFactory,
|
|
15
|
+
)
|
|
16
|
+
from wbcore.tests.conftest import *
|
|
17
|
+
|
|
18
|
+
register(UserFactory)
|
|
19
|
+
register(PersonFactory)
|
|
20
|
+
register(NotificationTypeModelFactory, name="notification_type")
|
|
21
|
+
register(NotificationTypeSettingModelFactory, name="notification_type_setting")
|
|
22
|
+
register(NotificationModelFactory, name="notification")
|
|
23
|
+
register(NotificationUserTokenModelFactory, name="notification_user_token")
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
@pytest.fixture
|
|
27
|
+
def request_with_user(user):
|
|
28
|
+
request = APIRequestFactory().get("/")
|
|
29
|
+
request.user = user
|
|
30
|
+
return request
|
|
31
|
+
|
|
32
|
+
|
|
33
|
+
@pytest.fixture
|
|
34
|
+
def client():
|
|
35
|
+
return APIClient()
|
|
36
|
+
|
|
37
|
+
|
|
38
|
+
@pytest.fixture(
|
|
39
|
+
autouse=True, scope="session"
|
|
40
|
+
) # Might want to find a way to registered default conftest logic automatically
|
|
41
|
+
def django_test_environment(django_test_environment):
|
|
42
|
+
from django.apps import apps
|
|
43
|
+
|
|
44
|
+
get_models = apps.get_models
|
|
45
|
+
|
|
46
|
+
for m in [m for m in get_models() if not m._meta.managed]:
|
|
47
|
+
m._meta.managed = True
|
|
File without changes
|
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
import pytest
|
|
2
|
+
|
|
3
|
+
from wbcore.contrib.notifications.backends.firebase.backends import (
|
|
4
|
+
NotificationBackend as FirebaseNotificationBackend,
|
|
5
|
+
)
|
|
6
|
+
from wbcore.contrib.notifications.factories.tokens import (
|
|
7
|
+
NotificationUserTokenModelFactory,
|
|
8
|
+
)
|
|
9
|
+
from wbcore.contrib.notifications.models.tokens import NotificationUserToken
|
|
10
|
+
|
|
11
|
+
|
|
12
|
+
@pytest.mark.django_db
|
|
13
|
+
def test_send_notification(mocker, notification, user):
|
|
14
|
+
mocked_get_app = mocker.patch(
|
|
15
|
+
"wbcore.contrib.notifications.backends.firebase.backends.firebase_admin.get_app",
|
|
16
|
+
)
|
|
17
|
+
mocked_initialize_app = mocker.patch(
|
|
18
|
+
"wbcore.contrib.notifications.backends.firebase.backends.firebase_admin.initialize_app",
|
|
19
|
+
)
|
|
20
|
+
mocked_send = mocker.patch("wbcore.contrib.notifications.backends.firebase.backends.messaging.send")
|
|
21
|
+
mocked_get_firebase_credentials = mocker.patch.object(FirebaseNotificationBackend, "get_firebase_credentials")
|
|
22
|
+
|
|
23
|
+
NotificationUserTokenModelFactory.create(user=user, device_type=NotificationUserToken.NotificationDeviceType.WEB)
|
|
24
|
+
notification.notification_type.user_settings.filter(user=notification.user).update(enable_web=True)
|
|
25
|
+
FirebaseNotificationBackend.send_notification(notification)
|
|
26
|
+
|
|
27
|
+
mocked_send.assert_called_once()
|
|
28
|
+
mocked_get_firebase_credentials.assert_called_once()
|
|
29
|
+
mocked_get_app.assert_called_once()
|
|
30
|
+
mocked_initialize_app.assert_not_called()
|
|
31
|
+
|
|
32
|
+
|
|
33
|
+
@pytest.mark.django_db
|
|
34
|
+
def test_send_multiple_notifications(mocker, notification, user):
|
|
35
|
+
mocked_get_app = mocker.patch(
|
|
36
|
+
"wbcore.contrib.notifications.backends.firebase.backends.firebase_admin.get_app",
|
|
37
|
+
)
|
|
38
|
+
mocked_initialize_app = mocker.patch(
|
|
39
|
+
"wbcore.contrib.notifications.backends.firebase.backends.firebase_admin.initialize_app",
|
|
40
|
+
)
|
|
41
|
+
mocked_send = mocker.patch("wbcore.contrib.notifications.backends.firebase.backends.messaging.send")
|
|
42
|
+
mocked_get_firebase_credentials = mocker.patch.object(FirebaseNotificationBackend, "get_firebase_credentials")
|
|
43
|
+
|
|
44
|
+
NotificationUserTokenModelFactory.create_batch(
|
|
45
|
+
3, user=user, device_type=NotificationUserToken.NotificationDeviceType.WEB
|
|
46
|
+
)
|
|
47
|
+
notification.notification_type.user_settings.filter(user=notification.user).update(enable_web=True)
|
|
48
|
+
FirebaseNotificationBackend.send_notification(notification)
|
|
49
|
+
|
|
50
|
+
assert mocked_send.call_count == 3
|
|
51
|
+
mocked_get_firebase_credentials.assert_called_once()
|
|
52
|
+
mocked_get_app.assert_called_once()
|
|
53
|
+
mocked_initialize_app.assert_not_called()
|
|
54
|
+
|
|
55
|
+
|
|
56
|
+
@pytest.mark.django_db
|
|
57
|
+
def test_send_notification_with_initialize(mocker, notification, user):
|
|
58
|
+
mocked_get_app = mocker.patch(
|
|
59
|
+
"wbcore.contrib.notifications.backends.firebase.backends.firebase_admin.get_app",
|
|
60
|
+
side_effect=ValueError(),
|
|
61
|
+
)
|
|
62
|
+
mocked_initialize_app = mocker.patch(
|
|
63
|
+
"wbcore.contrib.notifications.backends.firebase.backends.firebase_admin.initialize_app"
|
|
64
|
+
)
|
|
65
|
+
mocked_send = mocker.patch("wbcore.contrib.notifications.backends.firebase.backends.messaging.send")
|
|
66
|
+
mocked_get_firebase_credentials = mocker.patch.object(FirebaseNotificationBackend, "get_firebase_credentials")
|
|
67
|
+
NotificationUserTokenModelFactory.create(user=user, device_type=NotificationUserToken.NotificationDeviceType.WEB)
|
|
68
|
+
notification.notification_type.user_settings.filter(user=notification.user).update(enable_web=True)
|
|
69
|
+
FirebaseNotificationBackend.send_notification(notification)
|
|
70
|
+
|
|
71
|
+
mocked_get_app.assert_called_once()
|
|
72
|
+
mocked_initialize_app.assert_called_once()
|
|
73
|
+
mocked_send.assert_called_once()
|
|
74
|
+
mocked_get_firebase_credentials.assert_called_once()
|
|
75
|
+
|
|
76
|
+
|
|
77
|
+
def test_get_configuration(mocker):
|
|
78
|
+
mocker.patch.dict("os.environ", {"FIREBASE_WEB_CONFIG": '{"ABC": "123"}', "FIREBASE_VAPID_KEY": "DEF"})
|
|
79
|
+
assert FirebaseNotificationBackend.get_configuration() == {"firebase_config": {"ABC": "123"}, "vapid_key": "DEF"}
|
|
File without changes
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
import pytest
|
|
2
|
+
from django.db.utils import IntegrityError
|
|
3
|
+
|
|
4
|
+
from wbcore.contrib.authentication.factories import UserFactory
|
|
5
|
+
from wbcore.contrib.notifications.factories.notification_types import (
|
|
6
|
+
NotificationTypeModelFactory,
|
|
7
|
+
)
|
|
8
|
+
from wbcore.contrib.notifications.models import (
|
|
9
|
+
NotificationType,
|
|
10
|
+
NotificationTypeSetting,
|
|
11
|
+
)
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
@pytest.mark.django_db
|
|
15
|
+
class TestNotificationType:
|
|
16
|
+
def test_factory(self, notification_type: NotificationType):
|
|
17
|
+
assert isinstance(notification_type, NotificationType)
|
|
18
|
+
assert notification_type.pk is not None
|
|
19
|
+
|
|
20
|
+
def test_to_str(self, notification_type: NotificationType):
|
|
21
|
+
assert str(notification_type) == notification_type.title
|
|
22
|
+
|
|
23
|
+
def test_endpoint_basename(self):
|
|
24
|
+
assert NotificationType.get_endpoint_basename() == "wbcore:notifications:notification_type"
|
|
25
|
+
|
|
26
|
+
def test_representation_endpoint(self):
|
|
27
|
+
assert (
|
|
28
|
+
NotificationType.get_representation_endpoint()
|
|
29
|
+
== "wbcore:notifications:notification_type_representation-list"
|
|
30
|
+
)
|
|
31
|
+
|
|
32
|
+
def test_representation_value_key(self):
|
|
33
|
+
assert NotificationType.get_representation_value_key() == "id"
|
|
34
|
+
|
|
35
|
+
def test_representation_label_key(self):
|
|
36
|
+
assert NotificationType.get_representation_label_key() == "{{title}}"
|
|
37
|
+
|
|
38
|
+
@pytest.mark.parametrize("notification_type__code", ["code"])
|
|
39
|
+
def test_unique_code(self, notification_type: NotificationType):
|
|
40
|
+
assert notification_type.pk is not None
|
|
41
|
+
|
|
42
|
+
with pytest.raises(IntegrityError):
|
|
43
|
+
NotificationTypeModelFactory(code="code")
|
|
44
|
+
|
|
45
|
+
|
|
46
|
+
@pytest.mark.django_db
|
|
47
|
+
class TestNotificationTypeSetting:
|
|
48
|
+
def test_factory(self, notification_type_setting: NotificationTypeSetting):
|
|
49
|
+
assert isinstance(notification_type_setting, NotificationTypeSetting)
|
|
50
|
+
assert notification_type_setting.pk is not None
|
|
51
|
+
|
|
52
|
+
def test_to_str(self, notification_type_setting: NotificationTypeSetting):
|
|
53
|
+
template = "{0.user}: {0.notification_type} ({0.enable_web}/{0.enable_mobile}/{0.enable_email})"
|
|
54
|
+
assert str(notification_type_setting) == template.format(notification_type_setting)
|
|
55
|
+
|
|
56
|
+
def test_endpoint_basename(self):
|
|
57
|
+
assert NotificationTypeSetting.get_endpoint_basename() == "wbcore:notifications:notification_type_setting"
|
|
58
|
+
|
|
59
|
+
def test_representation_endpoint(self):
|
|
60
|
+
assert (
|
|
61
|
+
NotificationTypeSetting.get_representation_endpoint()
|
|
62
|
+
== "wbcore:notifications:notification_type_setting_representation-list"
|
|
63
|
+
)
|
|
64
|
+
|
|
65
|
+
def test_representation_value_key(self):
|
|
66
|
+
assert NotificationTypeSetting.get_representation_value_key() == "id"
|
|
67
|
+
|
|
68
|
+
def test_representation_label_key(self):
|
|
69
|
+
assert NotificationTypeSetting.get_representation_label_key() == "{{notification_type}}"
|
|
70
|
+
|
|
71
|
+
def test_create_through_post_save_from_notification_type(self, user):
|
|
72
|
+
assert user.pk is not None
|
|
73
|
+
type_count = NotificationTypeSetting.objects.filter(user=user).count()
|
|
74
|
+
|
|
75
|
+
NotificationTypeModelFactory()
|
|
76
|
+
|
|
77
|
+
assert NotificationTypeSetting.objects.filter(user=user).count() == type_count + 1
|
|
78
|
+
|
|
79
|
+
def test_create_through_post_save_from_user(self, notification_type):
|
|
80
|
+
assert notification_type.pk is not None
|
|
81
|
+
assert not NotificationTypeSetting.objects.filter(notification_type=notification_type).exists()
|
|
82
|
+
|
|
83
|
+
UserFactory()
|
|
84
|
+
|
|
85
|
+
assert NotificationTypeSetting.objects.filter(notification_type=notification_type).exists()
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
import pytest
|
|
2
|
+
|
|
3
|
+
from wbcore.contrib.notifications.models import Notification
|
|
4
|
+
from wbcore.contrib.notifications.utils import base_domain
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
@pytest.mark.django_db
|
|
8
|
+
class TestNotification:
|
|
9
|
+
def test_factory(self, notification: Notification):
|
|
10
|
+
assert isinstance(notification, Notification)
|
|
11
|
+
assert notification.pk is not None
|
|
12
|
+
|
|
13
|
+
def test_to_str(self, notification: Notification):
|
|
14
|
+
assert str(notification) == f"{notification.user} {notification.title}"
|
|
15
|
+
|
|
16
|
+
def test_endpoint_basename(self):
|
|
17
|
+
assert Notification.get_endpoint_basename() == "wbcore:notifications:notification"
|
|
18
|
+
|
|
19
|
+
def test_representation_value_key(self):
|
|
20
|
+
assert Notification.get_representation_value_key() == "id"
|
|
21
|
+
|
|
22
|
+
def test_representation_label_key(self):
|
|
23
|
+
assert Notification.get_representation_label_key() == "{{title}}"
|
|
24
|
+
|
|
25
|
+
@pytest.mark.parametrize("notification__endpoint", ["/wbcore/notifications/"])
|
|
26
|
+
def test_full_valid_internal_endpoint(self, notification):
|
|
27
|
+
assert notification.get_full_endpoint() == f"{base_domain()}{notification.endpoint}"
|
|
28
|
+
|
|
29
|
+
@pytest.mark.parametrize("notification__endpoint", ["/some_invalid_namespace/notifications/"])
|
|
30
|
+
def test_full_invalid_internal_endpoint(self, notification):
|
|
31
|
+
assert notification.get_full_endpoint() is None
|
|
32
|
+
|
|
33
|
+
@pytest.mark.parametrize("notification__endpoint", ["/wbcore/notifications/"])
|
|
34
|
+
def test_full_internal_endpoint_as_shareable_link(self, notification):
|
|
35
|
+
assert (
|
|
36
|
+
notification.get_full_endpoint(as_shareable_internal_link=True)
|
|
37
|
+
== f"{base_domain()}?widget_endpoint={notification.endpoint}"
|
|
38
|
+
)
|
|
39
|
+
|
|
40
|
+
@pytest.mark.parametrize("notification__endpoint", ["https://www.google.com"])
|
|
41
|
+
def test_full_valid_external_endpoint(self, notification):
|
|
42
|
+
assert notification.get_full_endpoint() == notification.endpoint
|
|
43
|
+
|
|
44
|
+
@pytest.mark.parametrize("notification__endpoint", ["https.www.google.com"])
|
|
45
|
+
def test_full_invalid_external_endpoint(self, notification):
|
|
46
|
+
assert notification.get_full_endpoint() is None
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import pytest
|
|
2
|
+
|
|
3
|
+
from wbcore.contrib.notifications.factories.notification_types import (
|
|
4
|
+
NotificationTypeSettingModelFactory,
|
|
5
|
+
)
|
|
6
|
+
from wbcore.contrib.notifications.models import NotificationUserToken
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
@pytest.mark.django_db
|
|
10
|
+
class TestNotificationUserToken:
|
|
11
|
+
def test_factory(self, notification_user_token: NotificationUserToken):
|
|
12
|
+
assert isinstance(notification_user_token, NotificationUserToken)
|
|
13
|
+
assert notification_user_token.pk is not None
|
|
14
|
+
|
|
15
|
+
def test_to_str(self, notification_user_token: NotificationUserToken):
|
|
16
|
+
assert str(notification_user_token) == "{0.user}: {0.token} ({0.device_type})".format(notification_user_token)
|
|
17
|
+
|
|
18
|
+
def test_filter_for_settings(self, notification_user_token: NotificationUserToken):
|
|
19
|
+
setting = NotificationTypeSettingModelFactory(
|
|
20
|
+
user=notification_user_token.user, enable_web=True, enable_mobile=True
|
|
21
|
+
)
|
|
22
|
+
assert setting.enable_web is False
|
|
23
|
+
assert setting.enable_mobile is False
|
|
24
|
+
assert not NotificationUserToken.objects.filter_for_user_settings(setting).exists()
|
|
25
|
+
|
|
26
|
+
setting.enable_web = True
|
|
27
|
+
setting.enable_mobile = True
|
|
28
|
+
setting.save()
|
|
29
|
+
|
|
30
|
+
assert NotificationUserToken.objects.filter_for_user_settings(setting).first() == notification_user_token # type: ignore
|
|
File without changes
|