wbcore 2.2.1__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 +22 -0
- wbcore/cache/decorators.py +29 -0
- wbcore/cache/mixins.py +48 -0
- wbcore/cache/registry.py +90 -0
- wbcore/cache/views.py +18 -0
- wbcore/configs/__init__.py +11 -0
- wbcore/configs/configs.py +51 -0
- wbcore/configs/decorators.py +11 -0
- wbcore/configs/registry.py +35 -0
- wbcore/configs/views.py +11 -0
- wbcore/configurations/__init__.py +1 -0
- wbcore/configurations/base.py +45 -0
- wbcore/configurations/configurations/__init__.py +16 -0
- wbcore/configurations/configurations/apps.py +54 -0
- wbcore/configurations/configurations/authentication.py +43 -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 +10 -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 +68 -0
- wbcore/content_type/__init__.py +0 -0
- wbcore/content_type/admin.py +8 -0
- wbcore/content_type/filters.py +19 -0
- wbcore/content_type/serializers.py +89 -0
- wbcore/content_type/utils.py +30 -0
- wbcore/content_type/viewsets.py +80 -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 +13 -0
- wbcore/contrib/agenda/admin/conference_room.py +18 -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 +47 -0
- wbcore/contrib/agenda/factories/conference_room.py +19 -0
- wbcore/contrib/agenda/filters/__init__.py +2 -0
- wbcore/contrib/agenda/filters/calendar_item.py +64 -0
- wbcore/contrib/agenda/filters/conference_room.py +41 -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 +238 -0
- wbcore/contrib/agenda/models/conference_room.py +95 -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 +76 -0
- wbcore/contrib/agenda/serializers/conference_room.py +98 -0
- wbcore/contrib/agenda/signals.py +8 -0
- wbcore/contrib/agenda/tests/__init__.py +0 -0
- wbcore/contrib/agenda/tests/conftest.py +14 -0
- wbcore/contrib/agenda/tests/signals.py +16 -0
- wbcore/contrib/agenda/tests/test_models.py +34 -0
- wbcore/contrib/agenda/tests/test_viewsets.py +171 -0
- wbcore/contrib/agenda/tests/tests.py +25 -0
- wbcore/contrib/agenda/typings.py +19 -0
- wbcore/contrib/agenda/urls.py +25 -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 +20 -0
- wbcore/contrib/agenda/viewsets/calendar_items.py +168 -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 +41 -0
- wbcore/contrib/agenda/viewsets/endpoints/__init__.py +1 -0
- wbcore/contrib/agenda/viewsets/endpoints/calendar_items.py +18 -0
- wbcore/contrib/agenda/viewsets/menu/__init__.py +2 -0
- wbcore/contrib/agenda/viewsets/menu/calendar_items.py +17 -0
- wbcore/contrib/agenda/viewsets/menu/conference_room.py +37 -0
- wbcore/contrib/agenda/viewsets/titles/__init__.py +2 -0
- wbcore/contrib/agenda/viewsets/titles/calendar_items.py +7 -0
- wbcore/contrib/agenda/viewsets/titles/conference_room.py +24 -0
- wbcore/contrib/ai/__init__.py +0 -0
- wbcore/contrib/ai/apps.py +5 -0
- wbcore/contrib/ai/exceptions.py +42 -0
- wbcore/contrib/ai/llm/__init__.py +0 -0
- wbcore/contrib/ai/llm/config.py +107 -0
- wbcore/contrib/ai/llm/decorators.py +10 -0
- wbcore/contrib/ai/llm/mixins.py +35 -0
- wbcore/contrib/ai/llm/utils.py +45 -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 +162 -0
- wbcore/contrib/authentication/configs.py +9 -0
- wbcore/contrib/authentication/configurations.py +53 -0
- wbcore/contrib/authentication/dynamic_preferences_registry.py +26 -0
- wbcore/contrib/authentication/factories/__init__.py +3 -0
- wbcore/contrib/authentication/factories/tokens.py +15 -0
- wbcore/contrib/authentication/factories/users.py +82 -0
- wbcore/contrib/authentication/factories/users_activities.py +19 -0
- wbcore/contrib/authentication/filters.py +18 -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 +225 -0
- wbcore/contrib/authentication/models/users_activities.py +112 -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 +350 -0
- wbcore/contrib/authentication/serializers/users_activites.py +37 -0
- wbcore/contrib/authentication/tasks.py +29 -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 +17 -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 +33 -0
- wbcore/contrib/authentication/tests/test_tokens.py +126 -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 +97 -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 +55 -0
- wbcore/contrib/authentication/viewsets/display/__init__.py +6 -0
- wbcore/contrib/authentication/viewsets/display/user_activities.py +85 -0
- wbcore/contrib/authentication/viewsets/display/users.py +72 -0
- wbcore/contrib/authentication/viewsets/endpoints/__init__.py +6 -0
- wbcore/contrib/authentication/viewsets/endpoints/user_activities.py +22 -0
- wbcore/contrib/authentication/viewsets/endpoints/users.py +20 -0
- wbcore/contrib/authentication/viewsets/menu/__init__.py +6 -0
- wbcore/contrib/authentication/viewsets/menu/user_activities.py +39 -0
- wbcore/contrib/authentication/viewsets/menu/users.py +16 -0
- wbcore/contrib/authentication/viewsets/titles/__init__.py +12 -0
- wbcore/contrib/authentication/viewsets/titles/user_activities.py +31 -0
- wbcore/contrib/authentication/viewsets/titles/users.py +26 -0
- wbcore/contrib/authentication/viewsets/user_activities.py +222 -0
- wbcore/contrib/authentication/viewsets/users.py +328 -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/currency/__init__.py +0 -0
- wbcore/contrib/currency/admin.py +32 -0
- wbcore/contrib/currency/apps.py +5 -0
- wbcore/contrib/currency/dynamic_preferences_registry.py +39 -0
- wbcore/contrib/currency/factories.py +39 -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 +66 -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 +24 -0
- wbcore/contrib/currency/import_export/handlers/currency_fx_rates.py +28 -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 +176 -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 +40 -0
- wbcore/contrib/currency/tests/__init__.py +0 -0
- wbcore/contrib/currency/tests/conftest.py +7 -0
- wbcore/contrib/currency/tests/test_models.py +104 -0
- wbcore/contrib/currency/tests/test_serializers.py +62 -0
- wbcore/contrib/currency/tests/test_viewsets.py +197 -0
- wbcore/contrib/currency/urls.py +19 -0
- wbcore/contrib/currency/viewsets/__init__.py +2 -0
- wbcore/contrib/currency/viewsets/buttons/__init__.py +0 -0
- wbcore/contrib/currency/viewsets/currency.py +56 -0
- wbcore/contrib/currency/viewsets/currency_fx_rates.py +32 -0
- wbcore/contrib/currency/viewsets/display/__init__.py +2 -0
- wbcore/contrib/currency/viewsets/display/currency.py +37 -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 +14 -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 +24 -0
- wbcore/contrib/dataloader/dataloaders/proxies.py +81 -0
- wbcore/contrib/dataloader/models/__init__.py +1 -0
- wbcore/contrib/dataloader/models/entities.py +29 -0
- wbcore/contrib/dataloader/models/querysets/__init__.py +1 -0
- wbcore/contrib/dataloader/models/querysets/entities.py +27 -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 +6 -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 +10 -0
- wbcore/contrib/dataloader/tests/test_dataloaders.py +37 -0
- wbcore/contrib/dataloader/tests/test_entities.py +12 -0
- wbcore/contrib/dataloader/utils.py +17 -0
- wbcore/contrib/directory/__init__.py +0 -0
- wbcore/contrib/directory/admin/__init__.py +14 -0
- wbcore/contrib/directory/admin/contacts.py +54 -0
- wbcore/contrib/directory/admin/entries.py +230 -0
- wbcore/contrib/directory/admin/relationships.py +73 -0
- wbcore/contrib/directory/apps.py +6 -0
- wbcore/contrib/directory/configs.py +8 -0
- wbcore/contrib/directory/configurations.py +69 -0
- wbcore/contrib/directory/dynamic_preferences_registry.py +61 -0
- wbcore/contrib/directory/factories/__init__.py +34 -0
- wbcore/contrib/directory/factories/contacts.py +71 -0
- wbcore/contrib/directory/factories/entries.py +169 -0
- wbcore/contrib/directory/factories/relationships.py +77 -0
- wbcore/contrib/directory/filters/__init__.py +22 -0
- wbcore/contrib/directory/filters/contacts.py +233 -0
- wbcore/contrib/directory/filters/entries.py +247 -0
- wbcore/contrib/directory/filters/relationships.py +89 -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/__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 +889 -0
- wbcore/contrib/directory/models/relationships.py +634 -0
- wbcore/contrib/directory/preferences.py +10 -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 +63 -0
- wbcore/contrib/directory/serializers/companies.py +158 -0
- wbcore/contrib/directory/serializers/contacts.py +404 -0
- wbcore/contrib/directory/serializers/entries.py +344 -0
- wbcore/contrib/directory/serializers/entry_representations.py +35 -0
- wbcore/contrib/directory/serializers/persons.py +209 -0
- wbcore/contrib/directory/serializers/relationships.py +332 -0
- wbcore/contrib/directory/signals.py +3 -0
- wbcore/contrib/directory/tests/__init__.py +0 -0
- wbcore/contrib/directory/tests/conftest.py +60 -0
- wbcore/contrib/directory/tests/disable_signals.py +52 -0
- wbcore/contrib/directory/tests/e2e/__init__.py +7 -0
- wbcore/contrib/directory/tests/e2e/e2e_directory_utility.py +162 -0
- wbcore/contrib/directory/tests/signals.py +90 -0
- wbcore/contrib/directory/tests/test_configs.py +6 -0
- wbcore/contrib/directory/tests/test_filters.py +59 -0
- wbcore/contrib/directory/tests/test_models.py +285 -0
- wbcore/contrib/directory/tests/test_permissions.py +122 -0
- wbcore/contrib/directory/tests/test_serializers.py +201 -0
- wbcore/contrib/directory/tests/test_viewsets.py +677 -0
- wbcore/contrib/directory/tests/tests.py +130 -0
- wbcore/contrib/directory/typings.py +17 -0
- wbcore/contrib/directory/urls.py +135 -0
- wbcore/contrib/directory/viewsets/__init__.py +58 -0
- wbcore/contrib/directory/viewsets/buttons/__init__.py +7 -0
- wbcore/contrib/directory/viewsets/buttons/contacts.py +16 -0
- wbcore/contrib/directory/viewsets/buttons/entries.py +133 -0
- wbcore/contrib/directory/viewsets/buttons/relationships.py +31 -0
- wbcore/contrib/directory/viewsets/contacts.py +393 -0
- wbcore/contrib/directory/viewsets/display/__init__.py +36 -0
- wbcore/contrib/directory/viewsets/display/contacts.py +380 -0
- wbcore/contrib/directory/viewsets/display/entries.py +501 -0
- wbcore/contrib/directory/viewsets/display/relationships.py +315 -0
- wbcore/contrib/directory/viewsets/display/utils.py +33 -0
- wbcore/contrib/directory/viewsets/endpoints/__init__.py +22 -0
- wbcore/contrib/directory/viewsets/endpoints/contacts.py +54 -0
- wbcore/contrib/directory/viewsets/endpoints/entries.py +30 -0
- wbcore/contrib/directory/viewsets/endpoints/relationships.py +105 -0
- wbcore/contrib/directory/viewsets/entries.py +212 -0
- wbcore/contrib/directory/viewsets/menu/__init__.py +16 -0
- wbcore/contrib/directory/viewsets/menu/contacts.py +25 -0
- wbcore/contrib/directory/viewsets/menu/entries.py +61 -0
- wbcore/contrib/directory/viewsets/menu/relationships.py +31 -0
- wbcore/contrib/directory/viewsets/menu/utils.py +72 -0
- wbcore/contrib/directory/viewsets/mixins.py +11 -0
- wbcore/contrib/directory/viewsets/previews/__init__.py +2 -0
- wbcore/contrib/directory/viewsets/previews/contacts.py +18 -0
- wbcore/contrib/directory/viewsets/previews/entries.py +31 -0
- wbcore/contrib/directory/viewsets/relationships.py +241 -0
- wbcore/contrib/directory/viewsets/titles/__init__.py +30 -0
- wbcore/contrib/directory/viewsets/titles/contacts.py +122 -0
- wbcore/contrib/directory/viewsets/titles/entries.py +29 -0
- wbcore/contrib/directory/viewsets/titles/relationships.py +86 -0
- wbcore/contrib/directory/viewsets/titles/utils.py +46 -0
- wbcore/contrib/directory/viewsets/utils.py +101 -0
- wbcore/contrib/documents/__init__.py +0 -0
- wbcore/contrib/documents/admin.py +79 -0
- wbcore/contrib/documents/apps.py +6 -0
- wbcore/contrib/documents/factories.py +43 -0
- wbcore/contrib/documents/filters.py +82 -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 +57 -0
- wbcore/contrib/documents/models/document_types.py +35 -0
- wbcore/contrib/documents/models/documents.py +309 -0
- wbcore/contrib/documents/models/mixins.py +10 -0
- wbcore/contrib/documents/models/shareable_links.py +90 -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 +40 -0
- wbcore/contrib/documents/serializers/documents.py +64 -0
- wbcore/contrib/documents/serializers/shareable_links.py +90 -0
- wbcore/contrib/documents/urls.py +42 -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 +53 -0
- wbcore/contrib/documents/viewsets/buttons/shareable_links.py +20 -0
- wbcore/contrib/documents/viewsets/buttons/signals.py +13 -0
- wbcore/contrib/documents/viewsets/display/__init__.py +4 -0
- wbcore/contrib/documents/viewsets/display/document_model_relationships.py +18 -0
- wbcore/contrib/documents/viewsets/display/document_types.py +23 -0
- wbcore/contrib/documents/viewsets/display/documents.py +101 -0
- wbcore/contrib/documents/viewsets/display/shareable_links.py +65 -0
- wbcore/contrib/documents/viewsets/document_model_relationships.py +25 -0
- wbcore/contrib/documents/viewsets/document_types.py +37 -0
- wbcore/contrib/documents/viewsets/documents.py +107 -0
- wbcore/contrib/documents/viewsets/endpoints/__init__.py +5 -0
- wbcore/contrib/documents/viewsets/endpoints/documents.py +19 -0
- wbcore/contrib/documents/viewsets/endpoints/documents_model_relationships.py +13 -0
- wbcore/contrib/documents/viewsets/endpoints/shareable_links.py +45 -0
- wbcore/contrib/documents/viewsets/menu/__init__.py +1 -0
- wbcore/contrib/documents/viewsets/menu/documents.py +24 -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 +109 -0
- wbcore/contrib/documents/viewsets/titles/__init__.py +3 -0
- wbcore/contrib/documents/viewsets/titles/document_types.py +13 -0
- wbcore/contrib/documents/viewsets/titles/documents.py +26 -0
- wbcore/contrib/documents/viewsets/titles/shareable_links.py +13 -0
- wbcore/contrib/dynamic_preferences/__init__.py +0 -0
- wbcore/contrib/dynamic_preferences/types.py +19 -0
- wbcore/contrib/example_app/__init__.py +0 -0
- wbcore/contrib/example_app/admin.py +117 -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 +26 -0
- wbcore/contrib/example_app/factories/league.py +15 -0
- wbcore/contrib/example_app/factories/match.py +25 -0
- wbcore/contrib/example_app/factories/person.py +32 -0
- wbcore/contrib/example_app/factories/role.py +9 -0
- wbcore/contrib/example_app/factories/sport.py +12 -0
- wbcore/contrib/example_app/factories/stadium.py +12 -0
- wbcore/contrib/example_app/factories/team.py +35 -0
- wbcore/contrib/example_app/filters/__init__.py +9 -0
- wbcore/contrib/example_app/filters/event.py +80 -0
- wbcore/contrib/example_app/filters/league.py +74 -0
- wbcore/contrib/example_app/filters/match.py +92 -0
- wbcore/contrib/example_app/filters/person.py +77 -0
- wbcore/contrib/example_app/filters/role.py +17 -0
- wbcore/contrib/example_app/filters/sport.py +26 -0
- wbcore/contrib/example_app/filters/stadium.py +30 -0
- wbcore/contrib/example_app/filters/team.py +77 -0
- wbcore/contrib/example_app/filters/teamresult.py +47 -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 +906 -0
- wbcore/contrib/example_app/serializers/__init__.py +31 -0
- wbcore/contrib/example_app/serializers/league.py +111 -0
- wbcore/contrib/example_app/serializers/match_event.py +304 -0
- wbcore/contrib/example_app/serializers/person_team.py +280 -0
- wbcore/contrib/example_app/serializers/role.py +16 -0
- wbcore/contrib/example_app/serializers/season.py +53 -0
- wbcore/contrib/example_app/serializers/sport.py +37 -0
- wbcore/contrib/example_app/serializers/stadium.py +58 -0
- wbcore/contrib/example_app/serializers/teamresult.py +43 -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 +51 -0
- wbcore/contrib/example_app/tests/e2e/test_league.py +69 -0
- wbcore/contrib/example_app/tests/e2e/test_person.py +67 -0
- wbcore/contrib/example_app/tests/e2e/test_teams.py +56 -0
- wbcore/contrib/example_app/tests/signals.py +7 -0
- wbcore/contrib/example_app/tests/test_displays.py +40 -0
- wbcore/contrib/example_app/tests/test_filters.py +70 -0
- wbcore/contrib/example_app/tests/test_utils.py +25 -0
- wbcore/contrib/example_app/urls.py +79 -0
- wbcore/contrib/example_app/utils.py +20 -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 +18 -0
- wbcore/contrib/example_app/viewsets/buttons/team.py +23 -0
- wbcore/contrib/example_app/viewsets/displays/__init__.py +23 -0
- wbcore/contrib/example_app/viewsets/displays/event.py +186 -0
- wbcore/contrib/example_app/viewsets/displays/league.py +390 -0
- wbcore/contrib/example_app/viewsets/displays/match.py +264 -0
- wbcore/contrib/example_app/viewsets/displays/person.py +251 -0
- wbcore/contrib/example_app/viewsets/displays/role.py +16 -0
- wbcore/contrib/example_app/viewsets/displays/season.py +65 -0
- wbcore/contrib/example_app/viewsets/displays/sport.py +115 -0
- wbcore/contrib/example_app/viewsets/displays/stadium.py +149 -0
- wbcore/contrib/example_app/viewsets/displays/team.py +230 -0
- wbcore/contrib/example_app/viewsets/displays/teamresult.py +30 -0
- wbcore/contrib/example_app/viewsets/endpoints/__init__.py +12 -0
- wbcore/contrib/example_app/viewsets/endpoints/endpoints.py +79 -0
- wbcore/contrib/example_app/viewsets/event.py +287 -0
- wbcore/contrib/example_app/viewsets/league.py +72 -0
- wbcore/contrib/example_app/viewsets/match.py +119 -0
- wbcore/contrib/example_app/viewsets/menu/__init__.py +1 -0
- wbcore/contrib/example_app/viewsets/menu/menus.py +102 -0
- wbcore/contrib/example_app/viewsets/menus.py +63 -0
- wbcore/contrib/example_app/viewsets/person.py +133 -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 +30 -0
- wbcore/contrib/example_app/viewsets/team.py +68 -0
- wbcore/contrib/example_app/viewsets/teamresult.py +106 -0
- wbcore/contrib/example_app/viewsets/titles/__init__.py +9 -0
- wbcore/contrib/example_app/viewsets/titles/event.py +41 -0
- wbcore/contrib/example_app/viewsets/titles/league.py +24 -0
- wbcore/contrib/example_app/viewsets/titles/match.py +34 -0
- wbcore/contrib/example_app/viewsets/titles/person.py +35 -0
- wbcore/contrib/example_app/viewsets/titles/role.py +13 -0
- wbcore/contrib/example_app/viewsets/titles/sport.py +13 -0
- wbcore/contrib/example_app/viewsets/titles/stadium.py +13 -0
- wbcore/contrib/example_app/viewsets/titles/team.py +24 -0
- wbcore/contrib/example_app/viewsets/titles/teamresult.py +12 -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/import_export/__init__.py +0 -0
- wbcore/contrib/geography/import_export/resources/__init__.py +0 -0
- wbcore/contrib/geography/import_export/resources/geography.py +10 -0
- wbcore/contrib/geography/migrations/0001_initial.py +110 -0
- wbcore/contrib/geography/migrations/__init__.py +0 -0
- wbcore/contrib/geography/models.py +147 -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 +32 -0
- wbcore/contrib/geography/tests/__init__.py +0 -0
- wbcore/contrib/geography/tests/conftest.py +12 -0
- wbcore/contrib/geography/tests/signals.py +7 -0
- wbcore/contrib/geography/tests/test_models.py +12 -0
- wbcore/contrib/geography/tests/test_serializers.py +7 -0
- wbcore/contrib/geography/tests/test_viewsets.py +6 -0
- wbcore/contrib/geography/tests/tests.py +12 -0
- wbcore/contrib/geography/urls.py +12 -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 +57 -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 +11 -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 +7 -0
- wbcore/contrib/gleap/views.py +31 -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 +138 -0
- wbcore/contrib/guardian/models/models.py +29 -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 +93 -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/viewsets/__init__.py +1 -0
- wbcore/contrib/guardian/viewsets/configs/__init__.py +4 -0
- wbcore/contrib/guardian/viewsets/configs/buttons.py +27 -0
- wbcore/contrib/guardian/viewsets/configs/displays.py +43 -0
- wbcore/contrib/guardian/viewsets/configs/endpoints.py +18 -0
- wbcore/contrib/guardian/viewsets/configs/titles.py +16 -0
- wbcore/contrib/guardian/viewsets/mixins.py +6 -0
- wbcore/contrib/guardian/viewsets/viewsets.py +139 -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 +374 -0
- wbcore/contrib/icons/backends/material.py +135 -0
- wbcore/contrib/icons/icons.py +169 -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 +150 -0
- wbcore/contrib/io/apps.py +27 -0
- wbcore/contrib/io/backends/__init__.py +5 -0
- wbcore/contrib/io/backends/abstract.py +60 -0
- wbcore/contrib/io/backends/mail.py +85 -0
- wbcore/contrib/io/backends/sftp.py +63 -0
- wbcore/contrib/io/backends/stream.py +92 -0
- wbcore/contrib/io/backends/utils.py +42 -0
- wbcore/contrib/io/configs/__init__.py +0 -0
- wbcore/contrib/io/configs/endpoints.py +12 -0
- wbcore/contrib/io/configurations/__init__.py +1 -0
- wbcore/contrib/io/configurations/base.py +7 -0
- wbcore/contrib/io/dynamic_preferences_registry.py +15 -0
- wbcore/contrib/io/enums.py +15 -0
- wbcore/contrib/io/exceptions.py +16 -0
- wbcore/contrib/io/factories.py +202 -0
- wbcore/contrib/io/imports.py +305 -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/__init__.py +0 -0
- wbcore/contrib/io/mixins.py +37 -0
- wbcore/contrib/io/models.py +1005 -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 +177 -0
- wbcore/contrib/io/serializers.py +136 -0
- wbcore/contrib/io/tests/__init__.py +0 -0
- wbcore/contrib/io/tests/conftest.py +42 -0
- wbcore/contrib/io/tests/test_backends.py +128 -0
- wbcore/contrib/io/tests/test_exports.py +130 -0
- wbcore/contrib/io/tests/test_imports.py +167 -0
- wbcore/contrib/io/tests/test_models.py +363 -0
- wbcore/contrib/io/tests/tests.py +18 -0
- wbcore/contrib/io/urls.py +28 -0
- wbcore/contrib/io/utils.py +44 -0
- wbcore/contrib/io/viewset_mixins.py +301 -0
- wbcore/contrib/io/viewsets.py +138 -0
- wbcore/contrib/notifications/__init__.py +0 -0
- wbcore/contrib/notifications/admin.py +61 -0
- wbcore/contrib/notifications/apps.py +38 -0
- wbcore/contrib/notifications/backends/__init__.py +0 -0
- wbcore/contrib/notifications/backends/abstract_backend.py +13 -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 +102 -0
- wbcore/contrib/notifications/configs.py +12 -0
- wbcore/contrib/notifications/configurations.py +9 -0
- wbcore/contrib/notifications/dispatch.py +71 -0
- wbcore/contrib/notifications/factories/__init__.py +0 -0
- wbcore/contrib/notifications/factories/notification_types.py +22 -0
- wbcore/contrib/notifications/factories/notifications.py +16 -0
- wbcore/contrib/notifications/factories/tokens.py +11 -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/__init__.py +0 -0
- wbcore/contrib/notifications/models/__init__.py +3 -0
- wbcore/contrib/notifications/models/notification_types.py +112 -0
- wbcore/contrib/notifications/models/notifications.py +81 -0
- wbcore/contrib/notifications/models/tokens.py +45 -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 +36 -0
- wbcore/contrib/notifications/serializers/notifications.py +33 -0
- wbcore/contrib/notifications/tasks.py +55 -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 +78 -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 +84 -0
- wbcore/contrib/notifications/tests/test_models/test_notifications.py +45 -0
- wbcore/contrib/notifications/tests/test_models/test_tokens.py +21 -0
- wbcore/contrib/notifications/tests/test_serializers/__init__.py +0 -0
- wbcore/contrib/notifications/tests/test_serializers/test_notification_types.py +53 -0
- wbcore/contrib/notifications/tests/test_serializers/test_notifications.py +23 -0
- wbcore/contrib/notifications/tests/test_tasks.py +71 -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 +121 -0
- wbcore/contrib/notifications/tests/test_viewsets/test_notifications.py +120 -0
- wbcore/contrib/notifications/urls.py +26 -0
- wbcore/contrib/notifications/utils.py +20 -0
- wbcore/contrib/notifications/views.py +62 -0
- wbcore/contrib/notifications/viewsets/__init__.py +5 -0
- wbcore/contrib/notifications/viewsets/menus.py +14 -0
- wbcore/contrib/notifications/viewsets/notification_types.py +40 -0
- wbcore/contrib/notifications/viewsets/notifications.py +57 -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 +27 -0
- wbcore/contrib/tags/filters.py +42 -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 +103 -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 +87 -0
- wbcore/contrib/tags/signals.py +17 -0
- wbcore/contrib/tags/tests/__init__.py +0 -0
- wbcore/contrib/tags/tests/conftest.py +7 -0
- wbcore/contrib/tags/tests/tests.py +17 -0
- wbcore/contrib/tags/urls.py +14 -0
- wbcore/contrib/tags/viewsets/__init__.py +8 -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 +8 -0
- wbcore/contrib/workflow/admin/data.py +14 -0
- wbcore/contrib/workflow/admin/display.py +8 -0
- wbcore/contrib/workflow/admin/process.py +36 -0
- wbcore/contrib/workflow/admin/step.py +91 -0
- wbcore/contrib/workflow/admin/transition.py +8 -0
- wbcore/contrib/workflow/admin/workflow.py +8 -0
- wbcore/contrib/workflow/apps.py +26 -0
- wbcore/contrib/workflow/configs.py +10 -0
- wbcore/contrib/workflow/decorators.py +25 -0
- wbcore/contrib/workflow/dispatch.py +23 -0
- wbcore/contrib/workflow/factories/__init__.py +18 -0
- wbcore/contrib/workflow/factories/condition.py +15 -0
- wbcore/contrib/workflow/factories/data.py +22 -0
- wbcore/contrib/workflow/factories/display.py +27 -0
- wbcore/contrib/workflow/factories/process.py +69 -0
- wbcore/contrib/workflow/factories/step.py +155 -0
- wbcore/contrib/workflow/factories/transition.py +25 -0
- wbcore/contrib/workflow/factories/workflow.py +19 -0
- wbcore/contrib/workflow/filters/__init__.py +24 -0
- wbcore/contrib/workflow/filters/condition.py +23 -0
- wbcore/contrib/workflow/filters/data.py +24 -0
- wbcore/contrib/workflow/filters/process.py +164 -0
- wbcore/contrib/workflow/filters/step.py +226 -0
- wbcore/contrib/workflow/filters/transition.py +41 -0
- wbcore/contrib/workflow/filters/workflow.py +43 -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 +123 -0
- wbcore/contrib/workflow/models/data.py +238 -0
- wbcore/contrib/workflow/models/display.py +33 -0
- wbcore/contrib/workflow/models/process.py +243 -0
- wbcore/contrib/workflow/models/step.py +735 -0
- wbcore/contrib/workflow/models/transition.py +70 -0
- wbcore/contrib/workflow/models/workflow.py +307 -0
- wbcore/contrib/workflow/serializers/__init__.py +37 -0
- wbcore/contrib/workflow/serializers/condition.py +64 -0
- wbcore/contrib/workflow/serializers/data.py +135 -0
- wbcore/contrib/workflow/serializers/display.py +25 -0
- wbcore/contrib/workflow/serializers/process.py +182 -0
- wbcore/contrib/workflow/serializers/signals.py +25 -0
- wbcore/contrib/workflow/serializers/step.py +364 -0
- wbcore/contrib/workflow/serializers/transition.py +80 -0
- wbcore/contrib/workflow/serializers/workflow.py +124 -0
- wbcore/contrib/workflow/sites.py +43 -0
- wbcore/contrib/workflow/tests/__init__.py +0 -0
- wbcore/contrib/workflow/tests/conftest.py +57 -0
- wbcore/contrib/workflow/tests/test_configs.py +6 -0
- wbcore/contrib/workflow/tests/test_dispatch.py +88 -0
- wbcore/contrib/workflow/tests/test_displays.py +119 -0
- wbcore/contrib/workflow/tests/test_filters.py +86 -0
- wbcore/contrib/workflow/tests/test_serializers.py +177 -0
- wbcore/contrib/workflow/tests/test_viewsets.py +358 -0
- wbcore/contrib/workflow/tests/test_workflow_assignees.py +225 -0
- wbcore/contrib/workflow/tests/tests.py +24 -0
- wbcore/contrib/workflow/urls.py +66 -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 +67 -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 +50 -0
- wbcore/contrib/workflow/viewsets/display/data.py +50 -0
- wbcore/contrib/workflow/viewsets/display/process.py +185 -0
- wbcore/contrib/workflow/viewsets/display/step.py +453 -0
- wbcore/contrib/workflow/viewsets/display/transition.py +75 -0
- wbcore/contrib/workflow/viewsets/display/workflow.py +167 -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 +18 -0
- wbcore/contrib/workflow/viewsets/menu/__init__.py +15 -0
- wbcore/contrib/workflow/viewsets/menu/condition.py +18 -0
- wbcore/contrib/workflow/viewsets/menu/data.py +18 -0
- wbcore/contrib/workflow/viewsets/menu/process.py +18 -0
- wbcore/contrib/workflow/viewsets/menu/step.py +123 -0
- wbcore/contrib/workflow/viewsets/menu/transition.py +18 -0
- wbcore/contrib/workflow/viewsets/menu/workflow.py +18 -0
- wbcore/contrib/workflow/viewsets/process.py +171 -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 +13 -0
- wbcore/contrib/workflow/viewsets/titles/data.py +13 -0
- wbcore/contrib/workflow/viewsets/titles/process.py +26 -0
- wbcore/contrib/workflow/viewsets/titles/step.py +101 -0
- wbcore/contrib/workflow/viewsets/titles/transition.py +13 -0
- wbcore/contrib/workflow/viewsets/titles/workflow.py +13 -0
- wbcore/contrib/workflow/viewsets/transition.py +57 -0
- wbcore/contrib/workflow/viewsets/workflow.py +36 -0
- wbcore/contrib/workflow/workflows/__init__.py +1 -0
- wbcore/contrib/workflow/workflows/assignees.py +82 -0
- wbcore/crontab/__init__.py +0 -0
- wbcore/crontab/serializers.py +23 -0
- wbcore/crontab/viewsets.py +9 -0
- wbcore/dispatch.py +55 -0
- wbcore/docs/__init__.py +25 -0
- wbcore/docs/orderable.md +29 -0
- wbcore/docs/reparent.md +13 -0
- wbcore/dynamic_preferences_registry.py +29 -0
- wbcore/enums.py +98 -0
- wbcore/filters/__init__.py +21 -0
- wbcore/filters/backends.py +19 -0
- wbcore/filters/defaults.py +69 -0
- wbcore/filters/fields/__init__.py +15 -0
- wbcore/filters/fields/booleans.py +6 -0
- wbcore/filters/fields/choices.py +61 -0
- wbcore/filters/fields/content_type.py +33 -0
- wbcore/filters/fields/datetime.py +91 -0
- wbcore/filters/fields/models.py +113 -0
- wbcore/filters/fields/multiple_lookups.py +19 -0
- wbcore/filters/fields/numbers.py +46 -0
- wbcore/filters/fields/text.py +6 -0
- wbcore/filters/filterset.py +217 -0
- wbcore/filters/lookups.py +41 -0
- wbcore/filters/mixins.py +112 -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 +114 -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 +53 -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 +38 -0
- wbcore/markdown/template.py +38 -0
- wbcore/markdown/utils.py +36 -0
- wbcore/markdown/views.py +67 -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 +51 -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 +91 -0
- wbcore/metadata/configs/buttons/buttons.py +142 -0
- wbcore/metadata/configs/buttons/enums.py +25 -0
- wbcore/metadata/configs/buttons/metadata.py +8 -0
- wbcore/metadata/configs/buttons/view_config.py +176 -0
- wbcore/metadata/configs/display/__init__.py +4 -0
- wbcore/metadata/configs/display/configs.py +15 -0
- wbcore/metadata/configs/display/display.py +234 -0
- wbcore/metadata/configs/display/formatting.py +50 -0
- wbcore/metadata/configs/display/instance_display/__init__.py +15 -0
- wbcore/metadata/configs/display/instance_display/display.py +37 -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 +70 -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 +230 -0
- wbcore/metadata/configs/display/models.py +20 -0
- wbcore/metadata/configs/display/view_config.py +82 -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 +176 -0
- wbcore/metadata/configs/fields.py +17 -0
- wbcore/metadata/configs/filter_fields.py +42 -0
- wbcore/metadata/configs/identifiers.py +25 -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 +33 -0
- wbcore/metadata/mixins.py +105 -0
- wbcore/metadata/tests/__init__.py +0 -0
- wbcore/metadata/tests/test_buttons.py +179 -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 +182 -0
- wbcore/models/fields.py +29 -0
- wbcore/models/orderable.py +6 -0
- wbcore/pagination.py +65 -0
- wbcore/pandas/__init__.py +0 -0
- wbcore/pandas/fields.py +136 -0
- wbcore/pandas/filters.py +113 -0
- wbcore/pandas/filterset.py +26 -0
- wbcore/pandas/metadata.py +14 -0
- wbcore/pandas/utils.py +129 -0
- wbcore/pandas/views.py +157 -0
- wbcore/permissions/__init__.py +0 -0
- wbcore/permissions/backend.py +35 -0
- wbcore/permissions/mixins.py +72 -0
- wbcore/permissions/permissions.py +50 -0
- wbcore/permissions/registry.py +32 -0
- wbcore/permissions/shortcuts.py +37 -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 +25 -0
- wbcore/release_notes/display.py +57 -0
- wbcore/release_notes/filters.py +35 -0
- wbcore/release_notes/models.py +65 -0
- wbcore/release_notes/serializers.py +19 -0
- wbcore/release_notes/utils.py +14 -0
- wbcore/release_notes/viewsets.py +45 -0
- wbcore/reversion/__init__.py +0 -0
- wbcore/reversion/filters.py +38 -0
- wbcore/reversion/serializers.py +80 -0
- wbcore/reversion/viewsets/__init__.py +6 -0
- wbcore/reversion/viewsets/buttons.py +94 -0
- wbcore/reversion/viewsets/displays.py +44 -0
- wbcore/reversion/viewsets/endpoints.py +29 -0
- wbcore/reversion/viewsets/titles.py +17 -0
- wbcore/reversion/viewsets/viewsets.py +111 -0
- wbcore/routers.py +63 -0
- wbcore/search/__init__.py +62 -0
- wbcore/serializers/__init__.py +66 -0
- wbcore/serializers/fields/__init__.py +58 -0
- wbcore/serializers/fields/boolean.py +51 -0
- wbcore/serializers/fields/choice.py +36 -0
- wbcore/serializers/fields/datetime.py +143 -0
- wbcore/serializers/fields/fields.py +190 -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 +105 -0
- wbcore/serializers/fields/mixins.py +175 -0
- wbcore/serializers/fields/number.py +103 -0
- wbcore/serializers/fields/other.py +42 -0
- wbcore/serializers/fields/primary_key.py +22 -0
- wbcore/serializers/fields/related.py +152 -0
- wbcore/serializers/fields/text.py +136 -0
- wbcore/serializers/fields/types.py +41 -0
- wbcore/serializers/mixins.py +24 -0
- wbcore/serializers/serializers.py +415 -0
- wbcore/serializers/utils.py +142 -0
- wbcore/shares/__init__.py +1 -0
- wbcore/shares/config.py +63 -0
- wbcore/shares/decorator.py +13 -0
- wbcore/shares/signals.py +3 -0
- wbcore/shares/sites.py +28 -0
- wbcore/shares/views.py +24 -0
- wbcore/signals/__init__.py +6 -0
- wbcore/signals/filters.py +3 -0
- wbcore/signals/instance_buttons.py +5 -0
- wbcore/signals/merge.py +4 -0
- wbcore/signals/models.py +8 -0
- wbcore/signals/permissions.py +3 -0
- wbcore/signals/serializers.py +5 -0
- wbcore/tasks.py +83 -0
- wbcore/templates/reversion/__init__.py +0 -0
- wbcore/templates/reversion/compare_detail.html +19 -0
- wbcore/test/__init__.py +30 -0
- wbcore/test/mixins.py +709 -0
- wbcore/test/signals.py +6 -0
- wbcore/test/tests.py +131 -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 +24 -0
- wbcore/tests/models.py +6 -0
- wbcore/tests/test_cache/__init__.py +0 -0
- wbcore/tests/test_cache/test_decorators.py +30 -0
- wbcore/tests/test_cache/test_mixins.py +29 -0
- wbcore/tests/test_cache/test_registry.py +57 -0
- wbcore/tests/test_configs.py +65 -0
- wbcore/tests/test_enums.py +54 -0
- wbcore/tests/test_fields/__init__.py +0 -0
- wbcore/tests/test_fields/test_boolean_fields.py +48 -0
- wbcore/tests/test_fields/test_choice_fields.py +48 -0
- wbcore/tests/test_fields/test_datetime_fields.py +151 -0
- wbcore/tests/test_fields/test_fields.py +23 -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 +111 -0
- wbcore/tests/test_fields/test_number_fields.py +190 -0
- wbcore/tests/test_fields/test_other_fields.py +61 -0
- wbcore/tests/test_fields/test_primary_key_fields.py +52 -0
- wbcore/tests/test_fields/test_related.py +80 -0
- wbcore/tests/test_fields/test_text_fields.py +117 -0
- wbcore/tests/test_filters/__init__.py +0 -0
- wbcore/tests/test_filters/test_mixins.py +108 -0
- wbcore/tests/test_filters/test_pandas.py +114 -0
- wbcore/tests/test_list_display.py +31 -0
- wbcore/tests/test_models/__init__.py +0 -0
- wbcore/tests/test_models/test_fields.py +25 -0
- wbcore/tests/test_models/test_mixins.py +31 -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 +15 -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 +48 -0
- wbcore/tests/test_pagination.py +31 -0
- wbcore/tests/test_serializers/__init__.py +0 -0
- wbcore/tests/test_serializers/test_fields.py +140 -0
- wbcore/tests/test_serializers/test_mixins.py +53 -0
- wbcore/tests/test_serializers/test_related.py +77 -0
- wbcore/tests/test_something.py +39 -0
- wbcore/tests/test_utils/__init__.py +0 -0
- wbcore/tests/test_utils/test_date.py +49 -0
- wbcore/tests/test_utils/test_date_builder.py +99 -0
- wbcore/tests/test_utils/test_primary.py +79 -0
- wbcore/tests/test_utils/test_signals.py +38 -0
- wbcore/tests/test_viewsets.py +20 -0
- wbcore/urls.py +114 -0
- wbcore/utils/__init__.py +13 -0
- wbcore/utils/cache.py +8 -0
- wbcore/utils/date.py +220 -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 +290 -0
- wbcore/utils/html.py +8 -0
- wbcore/utils/importlib.py +13 -0
- wbcore/utils/itertools.py +40 -0
- wbcore/utils/models.py +275 -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/rrules.py +37 -0
- wbcore/utils/serializers.py +8 -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/task.py +6 -0
- wbcore/utils/urls.py +70 -0
- wbcore/utils/views.py +201 -0
- wbcore/views.py +26 -0
- wbcore/viewsets/__init__.py +10 -0
- wbcore/viewsets/encoders.py +18 -0
- wbcore/viewsets/generics.py +5 -0
- wbcore/viewsets/mixins.py +290 -0
- wbcore/viewsets/utils.py +26 -0
- wbcore/viewsets/viewsets.py +142 -0
- wbcore-2.2.1.dist-info/METADATA +62 -0
- wbcore-2.2.1.dist-info/RECORD +1035 -0
- wbcore-2.2.1.dist-info/WHEEL +5 -0
|
@@ -0,0 +1,889 @@
|
|
|
1
|
+
import sys
|
|
2
|
+
import uuid
|
|
3
|
+
from contextlib import suppress
|
|
4
|
+
from pathlib import Path
|
|
5
|
+
|
|
6
|
+
from django.apps import apps
|
|
7
|
+
from django.conf import settings
|
|
8
|
+
from django.contrib.postgres.aggregates import StringAgg
|
|
9
|
+
from django.core.exceptions import AppRegistryNotReady
|
|
10
|
+
from django.core.validators import MaxValueValidator, MinValueValidator
|
|
11
|
+
from django.db import models
|
|
12
|
+
from django.db.models import CharField, Exists, F, OuterRef, Q, Subquery, Value
|
|
13
|
+
from django.db.models.functions import Concat
|
|
14
|
+
from django.db.models.signals import m2m_changed
|
|
15
|
+
from django.dispatch import receiver
|
|
16
|
+
from django.utils import timezone
|
|
17
|
+
from django.utils.functional import cached_property
|
|
18
|
+
from django.utils.translation import gettext_lazy as _
|
|
19
|
+
from dynamic_preferences.registries import global_preferences_registry
|
|
20
|
+
from slugify import slugify
|
|
21
|
+
from wbcore.contrib.agenda.models import CalendarItem
|
|
22
|
+
from wbcore.contrib.authentication.models import User
|
|
23
|
+
from wbcore.contrib.directory.signals import deactivate_profile
|
|
24
|
+
from wbcore.contrib.directory.typings import Person as PersonDTO
|
|
25
|
+
from wbcore.models import WBModel
|
|
26
|
+
from wbcore.permissions.shortcuts import get_internal_users
|
|
27
|
+
from wbcore.utils.models import (
|
|
28
|
+
ActiveObjectManager,
|
|
29
|
+
ComplexToStringMixin,
|
|
30
|
+
DeleteToDisableMixin,
|
|
31
|
+
)
|
|
32
|
+
|
|
33
|
+
from . import contacts
|
|
34
|
+
from .relationships import ClientManagerRelationship, EmployerEmployeeRelationship
|
|
35
|
+
|
|
36
|
+
|
|
37
|
+
class CustomerStatus(WBModel):
|
|
38
|
+
title = models.CharField(
|
|
39
|
+
max_length=32,
|
|
40
|
+
verbose_name=_("Title"),
|
|
41
|
+
unique=True,
|
|
42
|
+
blank=False,
|
|
43
|
+
null=False,
|
|
44
|
+
)
|
|
45
|
+
|
|
46
|
+
slugify_title = models.CharField(
|
|
47
|
+
max_length=32,
|
|
48
|
+
unique=True,
|
|
49
|
+
verbose_name="Slugified Title",
|
|
50
|
+
blank=True,
|
|
51
|
+
null=True,
|
|
52
|
+
)
|
|
53
|
+
|
|
54
|
+
def __str__(self) -> str:
|
|
55
|
+
return f"{self.title}"
|
|
56
|
+
|
|
57
|
+
@classmethod
|
|
58
|
+
def get_endpoint_basename(cls):
|
|
59
|
+
return "wbcore:directory:customerstatus"
|
|
60
|
+
|
|
61
|
+
@classmethod
|
|
62
|
+
def get_representation_value_key(cls):
|
|
63
|
+
return "id"
|
|
64
|
+
|
|
65
|
+
@classmethod
|
|
66
|
+
def get_representation_endpoint(cls):
|
|
67
|
+
return "wbcore:directory:customerstatusrepresentation-list"
|
|
68
|
+
|
|
69
|
+
@classmethod
|
|
70
|
+
def get_representation_label_key(cls):
|
|
71
|
+
return "{{title}}"
|
|
72
|
+
|
|
73
|
+
class Meta:
|
|
74
|
+
verbose_name = _("Customer Status")
|
|
75
|
+
verbose_name_plural = _("Customer Statuses")
|
|
76
|
+
|
|
77
|
+
def save(self, *args, **kwargs):
|
|
78
|
+
self.slugify_title = slugify(self.title, separator=" ")
|
|
79
|
+
super().save(*args, **kwargs)
|
|
80
|
+
|
|
81
|
+
|
|
82
|
+
class CompanyType(WBModel):
|
|
83
|
+
title = models.CharField(
|
|
84
|
+
max_length=128,
|
|
85
|
+
verbose_name=_("Title"),
|
|
86
|
+
unique=True,
|
|
87
|
+
blank=False,
|
|
88
|
+
null=False,
|
|
89
|
+
)
|
|
90
|
+
|
|
91
|
+
slugify_title = models.CharField(
|
|
92
|
+
max_length=128,
|
|
93
|
+
unique=True,
|
|
94
|
+
verbose_name="Slugified Title",
|
|
95
|
+
blank=True,
|
|
96
|
+
null=True,
|
|
97
|
+
)
|
|
98
|
+
|
|
99
|
+
def __str__(self) -> str:
|
|
100
|
+
return f"{self.title}"
|
|
101
|
+
|
|
102
|
+
@classmethod
|
|
103
|
+
def get_endpoint_basename(cls):
|
|
104
|
+
return "wbcore:directory:companytype"
|
|
105
|
+
|
|
106
|
+
@classmethod
|
|
107
|
+
def get_representation_value_key(cls):
|
|
108
|
+
return "id"
|
|
109
|
+
|
|
110
|
+
@classmethod
|
|
111
|
+
def get_representation_endpoint(cls):
|
|
112
|
+
return "wbcore:directory:companytyperepresentation-list"
|
|
113
|
+
|
|
114
|
+
@classmethod
|
|
115
|
+
def get_representation_label_key(cls):
|
|
116
|
+
return "{{title}}"
|
|
117
|
+
|
|
118
|
+
class Meta:
|
|
119
|
+
verbose_name = _("Company Type")
|
|
120
|
+
verbose_name_plural = _("Company Types")
|
|
121
|
+
|
|
122
|
+
def save(self, *args, **kwargs):
|
|
123
|
+
self.slugify_title = slugify(self.title, separator=" ")
|
|
124
|
+
super().save(*args, **kwargs)
|
|
125
|
+
|
|
126
|
+
|
|
127
|
+
class Specialization(WBModel):
|
|
128
|
+
title = models.CharField(
|
|
129
|
+
max_length=128,
|
|
130
|
+
verbose_name=_("Title"),
|
|
131
|
+
unique=True,
|
|
132
|
+
blank=False,
|
|
133
|
+
null=False,
|
|
134
|
+
)
|
|
135
|
+
|
|
136
|
+
slugify_title = models.CharField(
|
|
137
|
+
max_length=128,
|
|
138
|
+
unique=True,
|
|
139
|
+
verbose_name="Slugified Title",
|
|
140
|
+
blank=True,
|
|
141
|
+
null=True,
|
|
142
|
+
)
|
|
143
|
+
|
|
144
|
+
def __str__(self) -> str:
|
|
145
|
+
return f"{self.title}"
|
|
146
|
+
|
|
147
|
+
@classmethod
|
|
148
|
+
def get_endpoint_basename(cls):
|
|
149
|
+
return "wbcore:directory:specialization"
|
|
150
|
+
|
|
151
|
+
@classmethod
|
|
152
|
+
def get_representation_value_key(cls):
|
|
153
|
+
return "id"
|
|
154
|
+
|
|
155
|
+
@classmethod
|
|
156
|
+
def get_representation_endpoint(cls):
|
|
157
|
+
return "wbcore:directory:specializationrepresentation-list"
|
|
158
|
+
|
|
159
|
+
@classmethod
|
|
160
|
+
def get_representation_label_key(cls):
|
|
161
|
+
return "{{title}}"
|
|
162
|
+
|
|
163
|
+
class Meta:
|
|
164
|
+
verbose_name = _("Specialization")
|
|
165
|
+
verbose_name_plural = _("Specializations")
|
|
166
|
+
|
|
167
|
+
def save(self, *args, **kwargs):
|
|
168
|
+
self.slugify_title = slugify(self.title, separator=" ")
|
|
169
|
+
super().save(*args, **kwargs)
|
|
170
|
+
|
|
171
|
+
|
|
172
|
+
def upload_to_profile_images(instance, filename):
|
|
173
|
+
file_extension = "".join(Path(filename).suffixes)
|
|
174
|
+
return f"crm/entry/profile_images/{instance.uuid}{file_extension}"
|
|
175
|
+
|
|
176
|
+
|
|
177
|
+
def upload_to_signatures(instance, filename):
|
|
178
|
+
file_extension = "".join(Path(filename).suffixes)
|
|
179
|
+
return f"crm/entry/signatures/{instance.uuid}{file_extension}"
|
|
180
|
+
|
|
181
|
+
|
|
182
|
+
class EntryDefaultQueryset(models.QuerySet):
|
|
183
|
+
def filter_for_user(self, user: User) -> models.QuerySet:
|
|
184
|
+
"""
|
|
185
|
+
Protect the chained queryset and filter the entries that this user cannot see based on the following rules:
|
|
186
|
+
|
|
187
|
+
* internal user or supersuer: All
|
|
188
|
+
* external user: only itself, its colleagues and all its clients
|
|
189
|
+
* else: None
|
|
190
|
+
"""
|
|
191
|
+
profile = user.profile
|
|
192
|
+
if user.is_superuser or profile.is_internal:
|
|
193
|
+
return self
|
|
194
|
+
else:
|
|
195
|
+
return (
|
|
196
|
+
self.annotate(
|
|
197
|
+
has_valid_relationship_manager_role=Exists(
|
|
198
|
+
ClientManagerRelationship.objects.filter(
|
|
199
|
+
client__id=OuterRef("pk"), relationship_manager=profile
|
|
200
|
+
).exclude(status=ClientManagerRelationship.Status.REMOVED)
|
|
201
|
+
),
|
|
202
|
+
is_colleague=Exists(
|
|
203
|
+
EmployerEmployeeRelationship.objects.filter(
|
|
204
|
+
employee_id=OuterRef("pk"), employer__in=profile.employers.all()
|
|
205
|
+
)
|
|
206
|
+
),
|
|
207
|
+
)
|
|
208
|
+
.filter(
|
|
209
|
+
Q(is_colleague=True) # can see its colleagues
|
|
210
|
+
| Q(id__in=profile.employers.values("id")) # can see its colleagues
|
|
211
|
+
| Q(id=profile.id) # can see itself
|
|
212
|
+
| Q(
|
|
213
|
+
has_valid_relationship_manager_role=True
|
|
214
|
+
) # can see all entry where they have a valid relationship manager role
|
|
215
|
+
)
|
|
216
|
+
.distinct()
|
|
217
|
+
)
|
|
218
|
+
|
|
219
|
+
def filter_only_internal(self) -> models.QuerySet:
|
|
220
|
+
return self.filter(id__in=get_internal_users().values("profile"))
|
|
221
|
+
|
|
222
|
+
def annotate_all(self) -> models.QuerySet:
|
|
223
|
+
qs = self
|
|
224
|
+
with suppress(AppRegistryNotReady):
|
|
225
|
+
qs = qs.annotate(
|
|
226
|
+
primary_manager_repr=Subquery(
|
|
227
|
+
ClientManagerRelationship.objects.filter(
|
|
228
|
+
client__id=OuterRef("id"),
|
|
229
|
+
primary=True,
|
|
230
|
+
status__in=[
|
|
231
|
+
ClientManagerRelationship.Status.APPROVED,
|
|
232
|
+
ClientManagerRelationship.Status.PENDINGREMOVE,
|
|
233
|
+
],
|
|
234
|
+
).values("relationship_manager__computed_str")[:1]
|
|
235
|
+
),
|
|
236
|
+
primary_email=Subquery(
|
|
237
|
+
contacts.EmailContact.objects.filter(primary=True, entry__id=OuterRef("pk")).values("address")[:1],
|
|
238
|
+
output_field=CharField(),
|
|
239
|
+
),
|
|
240
|
+
secondary_email=Subquery(
|
|
241
|
+
contacts.EmailContact.objects.filter(primary=False, entry__id=OuterRef("pk")).values("address")[
|
|
242
|
+
:1
|
|
243
|
+
],
|
|
244
|
+
output_field=CharField(),
|
|
245
|
+
),
|
|
246
|
+
primary_telephone=Subquery(
|
|
247
|
+
contacts.TelephoneContact.objects.filter(primary=True, entry__id=OuterRef("pk")).values("number")[
|
|
248
|
+
:1
|
|
249
|
+
],
|
|
250
|
+
output_field=CharField(),
|
|
251
|
+
),
|
|
252
|
+
primary_address=Subquery(
|
|
253
|
+
contacts.AddressContact.objects.filter(primary=True, entry__id=OuterRef("pk"))
|
|
254
|
+
.annotate(
|
|
255
|
+
primary_address=Concat(
|
|
256
|
+
F("street"), Value(" "), F("zip"), Value(" "), F("geography_city__name")
|
|
257
|
+
)
|
|
258
|
+
)
|
|
259
|
+
.values("primary_address")[:1],
|
|
260
|
+
output_field=CharField(),
|
|
261
|
+
),
|
|
262
|
+
last_event=Subquery(
|
|
263
|
+
CalendarItem.objects.filter(
|
|
264
|
+
period__endswith__lte=timezone.now(),
|
|
265
|
+
visibility=CalendarItem.Visibility.PUBLIC,
|
|
266
|
+
entities=OuterRef("pk"),
|
|
267
|
+
)
|
|
268
|
+
.order_by("-period__startswith")
|
|
269
|
+
.values("title")[:1]
|
|
270
|
+
),
|
|
271
|
+
addresses_repr=StringAgg("addresses__city", delimiter=", "),
|
|
272
|
+
)
|
|
273
|
+
return qs
|
|
274
|
+
|
|
275
|
+
|
|
276
|
+
class EntryManager(ActiveObjectManager):
|
|
277
|
+
def get_queryset(self) -> EntryDefaultQueryset:
|
|
278
|
+
return EntryDefaultQueryset(self.model).filter(is_active=True)
|
|
279
|
+
|
|
280
|
+
def filter_for_user(self, user: User) -> models.QuerySet:
|
|
281
|
+
return self.get_queryset().filter_for_user(user)
|
|
282
|
+
|
|
283
|
+
def filter_only_internal(self) -> models.QuerySet:
|
|
284
|
+
return self.get_queryset().filter_only_internal()
|
|
285
|
+
|
|
286
|
+
def annotate_all(self):
|
|
287
|
+
return self.get_queryset().annotate_all()
|
|
288
|
+
|
|
289
|
+
|
|
290
|
+
class Entry(ComplexToStringMixin, DeleteToDisableMixin, WBModel):
|
|
291
|
+
AUTOMATICALLY_CLEAN_SOFT_DELETED_OBJECTS = True
|
|
292
|
+
|
|
293
|
+
class EntryType(models.TextChoices):
|
|
294
|
+
PERSON = "PERSON", _("Person")
|
|
295
|
+
COMPANY = "COMPANY", _("Company")
|
|
296
|
+
|
|
297
|
+
uuid = models.UUIDField(default=uuid.uuid4, editable=False, unique=True)
|
|
298
|
+
entry_type = models.CharField(max_length=255, default="", choices=EntryType.choices, verbose_name=_("Type"))
|
|
299
|
+
additional_fields = models.JSONField(default=dict, blank=True, null=True, verbose_name=_("Additional Fields"))
|
|
300
|
+
is_draft_entry = models.BooleanField(
|
|
301
|
+
default=False,
|
|
302
|
+
verbose_name=_("Draft Entry"),
|
|
303
|
+
help_text=_(
|
|
304
|
+
"Draft entries are entries that have been created automatically and still need verification by a human user."
|
|
305
|
+
),
|
|
306
|
+
)
|
|
307
|
+
relationship_managers = models.ManyToManyField(
|
|
308
|
+
"Person",
|
|
309
|
+
related_name="clients",
|
|
310
|
+
blank=True,
|
|
311
|
+
through="directory.ClientManagerRelationship",
|
|
312
|
+
through_fields=("client", "relationship_manager"),
|
|
313
|
+
verbose_name=_("Relationship Managers"),
|
|
314
|
+
help_text=_("People in charge of this entry"),
|
|
315
|
+
)
|
|
316
|
+
relationships = models.ManyToManyField(
|
|
317
|
+
blank=True,
|
|
318
|
+
symmetrical=False,
|
|
319
|
+
to="self",
|
|
320
|
+
through="directory.Relationship",
|
|
321
|
+
through_fields=("from_entry", "to_entry"),
|
|
322
|
+
verbose_name=_("The Entry's Relationships"),
|
|
323
|
+
)
|
|
324
|
+
# Activity heat describes how active a customer is. The more active a customer, the higher their activity heat.
|
|
325
|
+
activity_heat = models.FloatField(
|
|
326
|
+
validators=[MinValueValidator(0), MaxValueValidator(1)],
|
|
327
|
+
default=0,
|
|
328
|
+
verbose_name=_("Activity Heat"),
|
|
329
|
+
)
|
|
330
|
+
slugify_computed_str = models.CharField(
|
|
331
|
+
max_length=512, blank=True, null=True, verbose_name="Slugified Search Name"
|
|
332
|
+
)
|
|
333
|
+
profile_image = models.ImageField(blank=True, null=True, upload_to=upload_to_profile_images, max_length=256)
|
|
334
|
+
# TODO: Move to Person Model?
|
|
335
|
+
signature = models.ImageField(blank=True, null=True, upload_to=upload_to_signatures, max_length=256)
|
|
336
|
+
external_identifier = models.CharField(null=True, blank=True, max_length=128, unique=True)
|
|
337
|
+
salutation = models.CharField(max_length=255, verbose_name=_("Salutation"), null=True, blank=True)
|
|
338
|
+
|
|
339
|
+
class Meta:
|
|
340
|
+
verbose_name = _("Entry")
|
|
341
|
+
verbose_name_plural = _("Entries")
|
|
342
|
+
|
|
343
|
+
ordering = ("entry_type",)
|
|
344
|
+
|
|
345
|
+
objects = EntryManager()
|
|
346
|
+
|
|
347
|
+
def __str__(self):
|
|
348
|
+
return self.computed_str
|
|
349
|
+
|
|
350
|
+
@property
|
|
351
|
+
def is_company(self) -> bool:
|
|
352
|
+
return self.entry_type == Company.__name__
|
|
353
|
+
|
|
354
|
+
def primary_email_contact(self):
|
|
355
|
+
"""
|
|
356
|
+
Get primary email contact
|
|
357
|
+
"""
|
|
358
|
+
try:
|
|
359
|
+
return self.emails.get(primary=True)
|
|
360
|
+
except contacts.EmailContact.DoesNotExist:
|
|
361
|
+
return None
|
|
362
|
+
|
|
363
|
+
def primary_telephone_contact(self):
|
|
364
|
+
"""
|
|
365
|
+
Get primary Telephone contact
|
|
366
|
+
"""
|
|
367
|
+
try:
|
|
368
|
+
return self.telephones.get(primary=True)
|
|
369
|
+
except contacts.TelephoneContact.DoesNotExist:
|
|
370
|
+
return None
|
|
371
|
+
|
|
372
|
+
def primary_address_contact(self):
|
|
373
|
+
"""
|
|
374
|
+
Get primary address contact
|
|
375
|
+
"""
|
|
376
|
+
try:
|
|
377
|
+
return self.addresses.get(primary=True)
|
|
378
|
+
except contacts.AddressContact.DoesNotExist:
|
|
379
|
+
return None
|
|
380
|
+
|
|
381
|
+
def primary_website_contact(self):
|
|
382
|
+
"""
|
|
383
|
+
Get primary Website contact
|
|
384
|
+
"""
|
|
385
|
+
try:
|
|
386
|
+
return self.websites.get(primary=True)
|
|
387
|
+
except contacts.WebsiteContact.DoesNotExist:
|
|
388
|
+
return None
|
|
389
|
+
|
|
390
|
+
def primary_banking_contact(self):
|
|
391
|
+
"""
|
|
392
|
+
Get primary Banking contact
|
|
393
|
+
"""
|
|
394
|
+
try:
|
|
395
|
+
return self.banking.get(primary=True)
|
|
396
|
+
except contacts.BankingContact.DoesNotExist:
|
|
397
|
+
return None
|
|
398
|
+
|
|
399
|
+
def get_casted_entry(self):
|
|
400
|
+
"""
|
|
401
|
+
Cast the entry into its child representative. We use the all manager to be sure to get the casted object even after its "deletion"
|
|
402
|
+
"""
|
|
403
|
+
try:
|
|
404
|
+
model_class = getattr(sys.modules[__name__], self.entry_type)
|
|
405
|
+
except AttributeError:
|
|
406
|
+
model_class = self.__class__
|
|
407
|
+
return model_class.all_objects.get(pk=self.pk)
|
|
408
|
+
|
|
409
|
+
def compute_str(self):
|
|
410
|
+
self.slugify_computed_str = slugify(_("Entry {id}").format(id=self.id), separator=" ")
|
|
411
|
+
return _("Entry {id}").format(id=self.id)
|
|
412
|
+
|
|
413
|
+
def delete_additional_fields(self, additional_field_key=None):
|
|
414
|
+
if additional_field_key in self.additional_fields:
|
|
415
|
+
del self.additional_fields[additional_field_key]
|
|
416
|
+
|
|
417
|
+
@classmethod
|
|
418
|
+
def get_endpoint_basename(cls):
|
|
419
|
+
return "wbcore:directory:entry"
|
|
420
|
+
|
|
421
|
+
@classmethod
|
|
422
|
+
def get_representation_endpoint(cls):
|
|
423
|
+
return "wbcore:directory:entryrepresentation-list"
|
|
424
|
+
|
|
425
|
+
@classmethod
|
|
426
|
+
def get_representation_value_key(cls):
|
|
427
|
+
return "id"
|
|
428
|
+
|
|
429
|
+
|
|
430
|
+
class PersonDefaultQueryset(EntryDefaultQueryset):
|
|
431
|
+
def annotate_all(self) -> models.QuerySet:
|
|
432
|
+
qs = super().annotate_all()
|
|
433
|
+
with suppress(AppRegistryNotReady):
|
|
434
|
+
qs = qs.annotate(
|
|
435
|
+
primary_employer=Subquery(
|
|
436
|
+
EmployerEmployeeRelationship.objects.filter(employee__id=OuterRef("pk"), primary=True).values(
|
|
437
|
+
"employer__id"
|
|
438
|
+
)[:1]
|
|
439
|
+
),
|
|
440
|
+
primary_employer_repr=Subquery(
|
|
441
|
+
EmployerEmployeeRelationship.objects.filter(primary=True, employee=OuterRef("pk")).values(
|
|
442
|
+
"employer__computed_str"
|
|
443
|
+
)[:1]
|
|
444
|
+
),
|
|
445
|
+
primary_employer_number=Subquery(
|
|
446
|
+
contacts.TelephoneContact.objects.filter(
|
|
447
|
+
primary=True, entry__id=OuterRef("primary_employer")
|
|
448
|
+
).values("number")[:1],
|
|
449
|
+
output_field=CharField(),
|
|
450
|
+
),
|
|
451
|
+
name=Concat(F("first_name"), Value(" "), F("last_name")),
|
|
452
|
+
customer_status=Subquery(
|
|
453
|
+
EmployerEmployeeRelationship.objects.filter(primary=True, employee=OuterRef("pk")).values(
|
|
454
|
+
"employer__customer_status__title"
|
|
455
|
+
)[:1]
|
|
456
|
+
),
|
|
457
|
+
position_in_company=Subquery(
|
|
458
|
+
EmployerEmployeeRelationship.objects.filter(primary=True, employee=OuterRef("pk")).values(
|
|
459
|
+
"position__title"
|
|
460
|
+
)[:1]
|
|
461
|
+
),
|
|
462
|
+
tier=Subquery(
|
|
463
|
+
EmployerEmployeeRelationship.objects.filter(
|
|
464
|
+
primary=True,
|
|
465
|
+
employee=OuterRef("pk"),
|
|
466
|
+
).values(
|
|
467
|
+
"employer__tier"
|
|
468
|
+
)[:1]
|
|
469
|
+
),
|
|
470
|
+
)
|
|
471
|
+
return qs
|
|
472
|
+
|
|
473
|
+
|
|
474
|
+
class DefaultPersonManager(EntryManager):
|
|
475
|
+
def get_queryset(self):
|
|
476
|
+
qs = PersonDefaultQueryset(self.model).filter(is_active=True)
|
|
477
|
+
with suppress(Exception):
|
|
478
|
+
qs = qs.exclude(user_account__email=getattr(settings, "ANONYMOUS_USER_NAME", "AnonymousUser"))
|
|
479
|
+
return qs
|
|
480
|
+
|
|
481
|
+
|
|
482
|
+
class RegisteredPersonManager(DefaultPersonManager):
|
|
483
|
+
"""Custom Manager for filtering directly for registered user or non user"""
|
|
484
|
+
|
|
485
|
+
def get_queryset(self):
|
|
486
|
+
qs = super().get_queryset()
|
|
487
|
+
with suppress(Exception):
|
|
488
|
+
qs = qs.exclude(Q(user_account__isnull=False) & Q(user_account__is_register=False))
|
|
489
|
+
return qs
|
|
490
|
+
|
|
491
|
+
|
|
492
|
+
class PersonEmployeeManager(RegisteredPersonManager):
|
|
493
|
+
"""Custom Manager for filtering directly for Employees"""
|
|
494
|
+
|
|
495
|
+
def get_queryset(self):
|
|
496
|
+
if apps.is_installed("wbhuman_resources"):
|
|
497
|
+
return (
|
|
498
|
+
super()
|
|
499
|
+
.get_queryset()
|
|
500
|
+
.filter(
|
|
501
|
+
Q(human_resources__isnull=False)
|
|
502
|
+
& Q(human_resources__is_active=True)
|
|
503
|
+
& Q(user_account__isnull=False)
|
|
504
|
+
)
|
|
505
|
+
)
|
|
506
|
+
else:
|
|
507
|
+
try:
|
|
508
|
+
main_company_id = global_preferences_registry.manager()["directory__main_company"]
|
|
509
|
+
return super().get_queryset().filter(employers__id=main_company_id).distinct()
|
|
510
|
+
except Exception:
|
|
511
|
+
return super().get_queryset().none()
|
|
512
|
+
|
|
513
|
+
|
|
514
|
+
class Person(Entry):
|
|
515
|
+
class Prefix(models.TextChoices):
|
|
516
|
+
MR = "MR", _("Mr.")
|
|
517
|
+
MRS = "MRS", _("Mrs.")
|
|
518
|
+
PROF = "PROF", _("Prof.")
|
|
519
|
+
DR = "DR", _("Dr.")
|
|
520
|
+
MED = "MED", _("Med.")
|
|
521
|
+
ME = "ME", _("Me.")
|
|
522
|
+
|
|
523
|
+
"""Person Model that represents a person in the CRM"""
|
|
524
|
+
prefix = models.CharField(max_length=4, choices=Prefix.choices, verbose_name=_("Prefix"))
|
|
525
|
+
|
|
526
|
+
first_name = models.CharField(max_length=255, verbose_name=_("First Name"))
|
|
527
|
+
last_name = models.CharField(max_length=255, verbose_name=_("Last Name"))
|
|
528
|
+
initials = models.CharField(max_length=4, blank=True, null=True, verbose_name=_("Initials"))
|
|
529
|
+
|
|
530
|
+
employers = models.ManyToManyField(
|
|
531
|
+
"Company",
|
|
532
|
+
related_name="employees",
|
|
533
|
+
blank=True,
|
|
534
|
+
through="directory.EmployerEmployeeRelationship",
|
|
535
|
+
through_fields=("employee", "employer"),
|
|
536
|
+
verbose_name=_("Employers"),
|
|
537
|
+
help_text=_("The person's employers"),
|
|
538
|
+
)
|
|
539
|
+
|
|
540
|
+
active_employee = models.BooleanField(default=True, verbose_name=_("Available Employee"))
|
|
541
|
+
birthday = models.DateField(null=True, blank=True, verbose_name=_("Birthday"))
|
|
542
|
+
personality_profile_red = models.FloatField(
|
|
543
|
+
validators=[MinValueValidator(0), MaxValueValidator(1)],
|
|
544
|
+
default=0,
|
|
545
|
+
)
|
|
546
|
+
personality_profile_green = models.FloatField(
|
|
547
|
+
validators=[MinValueValidator(0), MaxValueValidator(1)],
|
|
548
|
+
default=0,
|
|
549
|
+
)
|
|
550
|
+
personality_profile_blue = models.FloatField(
|
|
551
|
+
validators=[MinValueValidator(0), MaxValueValidator(1)],
|
|
552
|
+
default=0,
|
|
553
|
+
)
|
|
554
|
+
|
|
555
|
+
formal = models.BooleanField(default=True, verbose_name=_("Formal"))
|
|
556
|
+
specializations = models.ManyToManyField(
|
|
557
|
+
to="Specialization",
|
|
558
|
+
blank=True,
|
|
559
|
+
verbose_name=_("Specializations"),
|
|
560
|
+
)
|
|
561
|
+
|
|
562
|
+
objects = DefaultPersonManager()
|
|
563
|
+
registered_users = RegisteredPersonManager()
|
|
564
|
+
|
|
565
|
+
class Meta:
|
|
566
|
+
verbose_name = _("Person")
|
|
567
|
+
verbose_name_plural = _("Persons")
|
|
568
|
+
|
|
569
|
+
@cached_property
|
|
570
|
+
def is_internal(self) -> bool:
|
|
571
|
+
if user := getattr(self, "user_account", None):
|
|
572
|
+
return user.is_internal
|
|
573
|
+
return False
|
|
574
|
+
|
|
575
|
+
@cached_property
|
|
576
|
+
def full_name(self):
|
|
577
|
+
return f"{self.last_name} {self.first_name}"
|
|
578
|
+
|
|
579
|
+
def __str__(self):
|
|
580
|
+
return f"{self.first_name} {self.last_name}"
|
|
581
|
+
|
|
582
|
+
def _build_dto(self):
|
|
583
|
+
return PersonDTO(
|
|
584
|
+
first_name=self.first_name,
|
|
585
|
+
last_name=self.last_name,
|
|
586
|
+
email=contact.address if (contact := self.primary_email_contact()) else None,
|
|
587
|
+
id=self.id,
|
|
588
|
+
)
|
|
589
|
+
|
|
590
|
+
def str_full(self):
|
|
591
|
+
"""
|
|
592
|
+
Get the string representation including Employers seperated title
|
|
593
|
+
"""
|
|
594
|
+
employers_repr = None
|
|
595
|
+
eer_rel = EmployerEmployeeRelationship.objects.filter(employee=self.id, primary=True)
|
|
596
|
+
try:
|
|
597
|
+
if eer_rel.exists():
|
|
598
|
+
employers_repr = Company.all_objects.get(id=eer_rel[0].employer.id).name
|
|
599
|
+
else:
|
|
600
|
+
employers_repr = (
|
|
601
|
+
"/".join(self.employers.all().values_list("name", flat=True)) if self.employers.exists() else ""
|
|
602
|
+
)
|
|
603
|
+
if employers_repr:
|
|
604
|
+
return f'{self.first_name} {self.last_name}{(" (%s)" % employers_repr)}'
|
|
605
|
+
except ValueError:
|
|
606
|
+
pass
|
|
607
|
+
return f"{self.first_name} {self.last_name}"
|
|
608
|
+
|
|
609
|
+
def compute_str(self):
|
|
610
|
+
self.slugify_computed_str = slugify(self.str_full(), separator=" ")
|
|
611
|
+
return self.str_full()
|
|
612
|
+
|
|
613
|
+
def get_initials(self):
|
|
614
|
+
initials = []
|
|
615
|
+
if self.first_name and len(self.first_name) > 1:
|
|
616
|
+
n = 1 if self.last_name else 2
|
|
617
|
+
initials.append(self.first_name.upper()[:n])
|
|
618
|
+
|
|
619
|
+
if self.last_name and len(self.last_name) > 1:
|
|
620
|
+
n = 1 if self.first_name else 2
|
|
621
|
+
initials.append(self.last_name.upper()[:n])
|
|
622
|
+
|
|
623
|
+
if len(initials) == 0:
|
|
624
|
+
initials.append("n.a.")
|
|
625
|
+
|
|
626
|
+
return "".join(initials)
|
|
627
|
+
|
|
628
|
+
def save(self, *args, **kwargs):
|
|
629
|
+
self.entry_type = "Person"
|
|
630
|
+
self.initials = self.get_initials()
|
|
631
|
+
if not self.salutation:
|
|
632
|
+
self.salutation = global_preferences_registry.manager()["directory__person_salutation"].format(
|
|
633
|
+
self.first_name, self.last_name
|
|
634
|
+
)
|
|
635
|
+
super().save(*args, **kwargs)
|
|
636
|
+
|
|
637
|
+
@classmethod
|
|
638
|
+
def get_or_create_with_user(cls, user, first_name="", last_name=""):
|
|
639
|
+
"""
|
|
640
|
+
Helper method that creates the CRM profile from user
|
|
641
|
+
|
|
642
|
+
Arguments:
|
|
643
|
+
user {User} -- The user used to generate the profile
|
|
644
|
+
"""
|
|
645
|
+
|
|
646
|
+
if profile := getattr(user, "profile", None):
|
|
647
|
+
return profile
|
|
648
|
+
else:
|
|
649
|
+
associated_emails = contacts.EmailContact.objects.filter(
|
|
650
|
+
address=user.email, entry__isnull=False, entry__entry_type="Person", entry__is_active=True
|
|
651
|
+
)
|
|
652
|
+
if (
|
|
653
|
+
associated_emails.exists()
|
|
654
|
+
and (person := Person.objects.get(id=associated_emails.first().entry.id))
|
|
655
|
+
and getattr(person, "user_account", None) is None
|
|
656
|
+
):
|
|
657
|
+
return person
|
|
658
|
+
else:
|
|
659
|
+
usernames = user.username.split("-")
|
|
660
|
+
if not first_name:
|
|
661
|
+
first_name = usernames[0]
|
|
662
|
+
if not last_name and len(usernames) > 1:
|
|
663
|
+
last_name = usernames[1]
|
|
664
|
+
person = cls.objects.create(first_name=first_name, last_name=last_name)
|
|
665
|
+
contacts.EmailContact.objects.create(
|
|
666
|
+
primary=True,
|
|
667
|
+
location=contacts.ContactLocationChoices.WORK.name,
|
|
668
|
+
entry=person,
|
|
669
|
+
address=user.email,
|
|
670
|
+
)
|
|
671
|
+
return person
|
|
672
|
+
|
|
673
|
+
@classmethod
|
|
674
|
+
def create_with_attributes(cls, first_name, last_name, email=None):
|
|
675
|
+
"""
|
|
676
|
+
Helper method that creates the CRM profile from user
|
|
677
|
+
|
|
678
|
+
Arguments:
|
|
679
|
+
user {User} -- The user used to generate the profile
|
|
680
|
+
"""
|
|
681
|
+
|
|
682
|
+
person = cls.objects.create(first_name=first_name, last_name=last_name)
|
|
683
|
+
if email:
|
|
684
|
+
contacts.EmailContact.objects.create(
|
|
685
|
+
primary=True,
|
|
686
|
+
location=contacts.ContactLocationChoices.WORK.name,
|
|
687
|
+
entry=person,
|
|
688
|
+
address=email,
|
|
689
|
+
)
|
|
690
|
+
|
|
691
|
+
return person
|
|
692
|
+
|
|
693
|
+
@classmethod
|
|
694
|
+
def get_endpoint_basename(cls):
|
|
695
|
+
return "wbcore:directory:person"
|
|
696
|
+
|
|
697
|
+
@classmethod
|
|
698
|
+
def get_representation_endpoint(cls):
|
|
699
|
+
return "wbcore:directory:personrepresentation-list"
|
|
700
|
+
|
|
701
|
+
@classmethod
|
|
702
|
+
def get_relationship_managers_representation_endpoint(cls):
|
|
703
|
+
return "wbcore:directory:personinchargerepresentation-list"
|
|
704
|
+
|
|
705
|
+
@classmethod
|
|
706
|
+
def get_representation_value_key(cls):
|
|
707
|
+
return "id"
|
|
708
|
+
|
|
709
|
+
|
|
710
|
+
class BankCompanyManager(EntryManager):
|
|
711
|
+
"""Custom Manager for filtering directly for Investment Banks"""
|
|
712
|
+
|
|
713
|
+
def get_queryset(self):
|
|
714
|
+
if hasattr(self, "issues_products"):
|
|
715
|
+
return super().get_queryset().filter(issues_products__isnull=False).distinct()
|
|
716
|
+
|
|
717
|
+
# This model manager is called from other places on startup time,
|
|
718
|
+
# hence we suppress the programming error if the migration has not yet run
|
|
719
|
+
return super().get_queryset().filter(type__title="Bank").distinct()
|
|
720
|
+
|
|
721
|
+
|
|
722
|
+
class Company(Entry):
|
|
723
|
+
class Tiering(models.TextChoices):
|
|
724
|
+
ONE = "ONE", "1"
|
|
725
|
+
TWO = "TWO", "2"
|
|
726
|
+
THREE = "THREE", "3"
|
|
727
|
+
FOUR = "FOUR", "4"
|
|
728
|
+
FIVE = "FIVE", "5"
|
|
729
|
+
|
|
730
|
+
"""
|
|
731
|
+
Company Model that represents a Company in the CRM
|
|
732
|
+
"""
|
|
733
|
+
|
|
734
|
+
type = models.ForeignKey(
|
|
735
|
+
"CompanyType",
|
|
736
|
+
related_name="company",
|
|
737
|
+
on_delete=models.SET_NULL,
|
|
738
|
+
verbose_name=_("Type"),
|
|
739
|
+
null=True,
|
|
740
|
+
blank=False,
|
|
741
|
+
)
|
|
742
|
+
|
|
743
|
+
name = models.CharField(max_length=255, verbose_name=_("Name"))
|
|
744
|
+
|
|
745
|
+
objects = EntryManager()
|
|
746
|
+
bank_objects = BankCompanyManager()
|
|
747
|
+
|
|
748
|
+
customer_status = models.ForeignKey(
|
|
749
|
+
"CustomerStatus",
|
|
750
|
+
related_name="company",
|
|
751
|
+
on_delete=models.SET_NULL,
|
|
752
|
+
verbose_name=_("Customer Status"),
|
|
753
|
+
null=True,
|
|
754
|
+
blank=True,
|
|
755
|
+
)
|
|
756
|
+
|
|
757
|
+
headcount = models.CharField(max_length=32, default="", blank=True, verbose_name=_("Number of employees"))
|
|
758
|
+
tier = models.CharField(max_length=16, null=True, blank=True, choices=Tiering.choices, verbose_name=_("Tier"))
|
|
759
|
+
|
|
760
|
+
description = models.TextField(default="", blank=True, verbose_name=_("Description"))
|
|
761
|
+
|
|
762
|
+
class Meta:
|
|
763
|
+
verbose_name = _("Company")
|
|
764
|
+
verbose_name_plural = _("Companies")
|
|
765
|
+
|
|
766
|
+
def __str__(self):
|
|
767
|
+
return f"{self.name}"
|
|
768
|
+
|
|
769
|
+
def save(self, *args, **kwargs):
|
|
770
|
+
self.entry_type = "Company"
|
|
771
|
+
if not self.salutation:
|
|
772
|
+
self.salutation = global_preferences_registry.manager()["directory__company_salutation"].format(self.name)
|
|
773
|
+
super().save(*args, **kwargs)
|
|
774
|
+
|
|
775
|
+
def compute_str(self):
|
|
776
|
+
self.slugify_computed_str = slugify(self.name, separator=" ")
|
|
777
|
+
return self.name
|
|
778
|
+
|
|
779
|
+
@classmethod
|
|
780
|
+
def get_endpoint_basename(cls):
|
|
781
|
+
return "wbcore:directory:company"
|
|
782
|
+
|
|
783
|
+
@classmethod
|
|
784
|
+
def get_representation_endpoint(cls):
|
|
785
|
+
return "wbcore:directory:companyrepresentation-list"
|
|
786
|
+
|
|
787
|
+
@classmethod
|
|
788
|
+
def get_representation_value_key(cls):
|
|
789
|
+
return "id"
|
|
790
|
+
|
|
791
|
+
|
|
792
|
+
@receiver(models.signals.post_save, sender=Entry)
|
|
793
|
+
def post_save_base_info(sender, instance, created, raw, **kwargs):
|
|
794
|
+
if not raw:
|
|
795
|
+
casted_entry = instance.get_casted_entry()
|
|
796
|
+
Entry.objects.filter(id=instance.id).update(
|
|
797
|
+
entry_type=casted_entry.__class__.__name__, computed_str=casted_entry.compute_str()
|
|
798
|
+
)
|
|
799
|
+
|
|
800
|
+
|
|
801
|
+
@receiver(models.signals.post_save, sender=Company)
|
|
802
|
+
def post_save_company(sender, instance, created, raw, **kwargs):
|
|
803
|
+
"""
|
|
804
|
+
Company post save signal: Triggers the post_save signals of all its employees, which updates their computed_str.
|
|
805
|
+
"""
|
|
806
|
+
if not raw:
|
|
807
|
+
if instance.employees.exists():
|
|
808
|
+
for employee in instance.employees.all():
|
|
809
|
+
employee.save()
|
|
810
|
+
|
|
811
|
+
|
|
812
|
+
@receiver(m2m_changed, sender=Person.employers.through)
|
|
813
|
+
def notify_entry_for_potential_claim(sender, instance, action, **kwargs):
|
|
814
|
+
"""
|
|
815
|
+
m2m change signal: If an employer is removed or added from the person employeers M2M, we need to recompute the computed_str
|
|
816
|
+
"""
|
|
817
|
+
if not instance.is_company:
|
|
818
|
+
instance.employers.all()
|
|
819
|
+
if action in ["post_add", "post_remove", "post_clear"]:
|
|
820
|
+
new_computed_str = instance.str_full()
|
|
821
|
+
Person.objects.filter(id=instance.id).update(computed_str=new_computed_str)
|
|
822
|
+
|
|
823
|
+
|
|
824
|
+
@receiver(deactivate_profile)
|
|
825
|
+
def handle_user_deactivation(sender, instance, substitute_profile=None, **kwargs):
|
|
826
|
+
messages = []
|
|
827
|
+
main_company_id = global_preferences_registry.manager()["directory__main_company"]
|
|
828
|
+
try:
|
|
829
|
+
main_company = Company.objects.get(id=main_company_id)
|
|
830
|
+
|
|
831
|
+
if main_company in instance.employers.all():
|
|
832
|
+
instance.employers.remove(main_company)
|
|
833
|
+
messages.append(
|
|
834
|
+
_("Removed {main_company} from {profile}'s employers").format(
|
|
835
|
+
main_company=main_company.computed_str, profile=instance.computed_str
|
|
836
|
+
)
|
|
837
|
+
)
|
|
838
|
+
except Company.DoesNotExist:
|
|
839
|
+
pass
|
|
840
|
+
users_clients = ClientManagerRelationship.objects.filter(relationship_manager=instance).exclude(
|
|
841
|
+
status=ClientManagerRelationship.Status.REMOVED
|
|
842
|
+
)
|
|
843
|
+
if users_clients.exists():
|
|
844
|
+
if substitute_profile:
|
|
845
|
+
client_number = users_clients.count()
|
|
846
|
+
for relationship in users_clients:
|
|
847
|
+
substitute_relationships = ClientManagerRelationship.objects.filter(
|
|
848
|
+
relationship_manager=substitute_profile, client=relationship.client
|
|
849
|
+
)
|
|
850
|
+
|
|
851
|
+
if relationship.status in [
|
|
852
|
+
ClientManagerRelationship.Status.DRAFT,
|
|
853
|
+
ClientManagerRelationship.Status.PENDINGADD,
|
|
854
|
+
ClientManagerRelationship.Status.PENDINGREMOVE,
|
|
855
|
+
]:
|
|
856
|
+
if substitute_relationships.exists():
|
|
857
|
+
relationship.delete()
|
|
858
|
+
else:
|
|
859
|
+
relationship.relationship_manager = substitute_profile
|
|
860
|
+
relationship.save()
|
|
861
|
+
elif relationship.status == ClientManagerRelationship.Status.APPROVED:
|
|
862
|
+
if substitute_relationships.exists():
|
|
863
|
+
for rel in substitute_relationships.all():
|
|
864
|
+
rel.primary = relationship.primary
|
|
865
|
+
rel.status = ClientManagerRelationship.Status.APPROVED
|
|
866
|
+
rel.save()
|
|
867
|
+
else:
|
|
868
|
+
ClientManagerRelationship.objects.create(
|
|
869
|
+
relationship_manager=substitute_profile,
|
|
870
|
+
client=relationship.client,
|
|
871
|
+
status=ClientManagerRelationship.Status.APPROVED,
|
|
872
|
+
primary=relationship.primary,
|
|
873
|
+
)
|
|
874
|
+
relationship.status = ClientManagerRelationship.Status.REMOVED
|
|
875
|
+
relationship.primary = False
|
|
876
|
+
relationship.save()
|
|
877
|
+
|
|
878
|
+
messages.append(
|
|
879
|
+
_("Assigned {clients} manager role(s) to {substitute_profile}").format(
|
|
880
|
+
clients=client_number, substitute_profile=substitute_profile.computed_str
|
|
881
|
+
)
|
|
882
|
+
)
|
|
883
|
+
|
|
884
|
+
else:
|
|
885
|
+
for rel in users_clients.all():
|
|
886
|
+
rel.status = ClientManagerRelationship.Status.REMOVED
|
|
887
|
+
rel.primary = False
|
|
888
|
+
rel.save()
|
|
889
|
+
return messages
|