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,91 @@
|
|
|
1
|
+
from contextlib import suppress
|
|
2
|
+
|
|
3
|
+
import django_filters
|
|
4
|
+
from wbcore.filters.mixins import WBCoreFilterMixin
|
|
5
|
+
from wbcore.forms import DateRangeField, DateTimeRangeField
|
|
6
|
+
from wbcore.utils.date import financial_performance_shortcuts
|
|
7
|
+
from wbcore.utils.date_builder.components import Component
|
|
8
|
+
|
|
9
|
+
|
|
10
|
+
class TimeFilter(WBCoreFilterMixin, django_filters.TimeFilter):
|
|
11
|
+
filter_type = "time"
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
class DateTimeFilter(WBCoreFilterMixin, django_filters.DateTimeFilter):
|
|
15
|
+
filter_type = "datetime"
|
|
16
|
+
|
|
17
|
+
|
|
18
|
+
class DateFilter(WBCoreFilterMixin, django_filters.DateFilter):
|
|
19
|
+
filter_type = "date"
|
|
20
|
+
|
|
21
|
+
|
|
22
|
+
class ShortcutAndPerformanceMixin(WBCoreFilterMixin):
|
|
23
|
+
def __init__(self, shortcuts: list | None = None, performance_mode: bool = False, *args, **kwargs):
|
|
24
|
+
self.shortcuts = shortcuts
|
|
25
|
+
self.performance_mode = performance_mode
|
|
26
|
+
super().__init__(*args, **kwargs)
|
|
27
|
+
|
|
28
|
+
def get_representation(self, request, name, view):
|
|
29
|
+
representation, lookup_expr = super().get_representation(request, name, view)
|
|
30
|
+
representation["performance_mode"] = self.performance_mode
|
|
31
|
+
|
|
32
|
+
if self.shortcuts:
|
|
33
|
+
representation["shortcuts"] = self.shortcuts
|
|
34
|
+
|
|
35
|
+
return representation, lookup_expr
|
|
36
|
+
|
|
37
|
+
|
|
38
|
+
class DateRangeFilter(ShortcutAndPerformanceMixin, django_filters.Filter):
|
|
39
|
+
field_class = DateRangeField
|
|
40
|
+
filter_type = "daterange"
|
|
41
|
+
default_format = "%Y-%m-%d"
|
|
42
|
+
|
|
43
|
+
def __init__(self, *args, **kwargs):
|
|
44
|
+
kwargs.setdefault("lookup_expr", "overlap")
|
|
45
|
+
super().__init__(*args, **kwargs)
|
|
46
|
+
|
|
47
|
+
def _get_default(self, *args):
|
|
48
|
+
default = super()._get_default(*args)
|
|
49
|
+
if default is not None:
|
|
50
|
+
lower = upper = None
|
|
51
|
+
if isinstance(default, tuple):
|
|
52
|
+
lower, upper = default
|
|
53
|
+
|
|
54
|
+
# if the default is a tuple of components, we need to convert them to string
|
|
55
|
+
if isinstance(lower, Component) and isinstance(upper, Component):
|
|
56
|
+
return f"{lower},{upper}"
|
|
57
|
+
|
|
58
|
+
elif hasattr(default, "lower") and hasattr(default, "upper"):
|
|
59
|
+
lower, upper = default.lower, default.upper
|
|
60
|
+
default = f'{lower.strftime(self.default_format) if lower else ""},{upper.strftime(self.default_format) if upper else ""}'
|
|
61
|
+
return default
|
|
62
|
+
|
|
63
|
+
def get_representation(self, request, name, view):
|
|
64
|
+
representation, lookup_expr = super().get_representation(request, name, view)
|
|
65
|
+
representation["lookup_expr"] = {"exact": self.field_name}
|
|
66
|
+
with suppress(KeyError): # TODO frontend needs to support both exact and overlaps lookup
|
|
67
|
+
default = representation["default"].pop(self.lookup_expr)
|
|
68
|
+
representation["default"]["exact"] = default
|
|
69
|
+
|
|
70
|
+
return representation, lookup_expr
|
|
71
|
+
|
|
72
|
+
@classmethod
|
|
73
|
+
def base_date_range_filter_method(cls, queryset, field_name, value):
|
|
74
|
+
if value:
|
|
75
|
+
filters = {}
|
|
76
|
+
if value.lower:
|
|
77
|
+
filters[f"{field_name}__gte"] = value.lower
|
|
78
|
+
if value.upper:
|
|
79
|
+
filters[f"{field_name}__lte"] = value.upper
|
|
80
|
+
return queryset.filter(**filters)
|
|
81
|
+
return queryset
|
|
82
|
+
|
|
83
|
+
|
|
84
|
+
class FinancialPerformanceDateRangeFilter(DateRangeFilter):
|
|
85
|
+
def __init__(self, *args, **kwargs):
|
|
86
|
+
super().__init__(performance_mode=True, shortcuts=financial_performance_shortcuts, *args, **kwargs)
|
|
87
|
+
|
|
88
|
+
|
|
89
|
+
class DateTimeRangeFilter(DateRangeFilter):
|
|
90
|
+
field_class = DateTimeRangeField
|
|
91
|
+
default_format = "%Y-%m-%dT%H:%M:%S%z"
|
|
@@ -0,0 +1,113 @@
|
|
|
1
|
+
from typing import Iterable
|
|
2
|
+
|
|
3
|
+
import django_filters
|
|
4
|
+
from django.core.exceptions import ObjectDoesNotExist
|
|
5
|
+
from django.forms.widgets import SelectMultiple, TextInput
|
|
6
|
+
from django.utils.http import urlencode
|
|
7
|
+
from django_filters.fields import ModelChoiceField, ModelMultipleChoiceField
|
|
8
|
+
from rest_framework.exceptions import ParseError
|
|
9
|
+
from rest_framework.reverse import reverse
|
|
10
|
+
from wbcore.filters.mixins import WBCoreFilterMixin
|
|
11
|
+
|
|
12
|
+
|
|
13
|
+
class ModelChoiceFilterMixin(WBCoreFilterMixin):
|
|
14
|
+
MULTIPLE: bool = False
|
|
15
|
+
|
|
16
|
+
@classmethod
|
|
17
|
+
def get_parsed_values(cls, queryset, value_ids: int | str | Iterable[int]):
|
|
18
|
+
if isinstance(value_ids, str):
|
|
19
|
+
value_ids = value_ids.split(",")
|
|
20
|
+
elif isinstance(value_ids, int):
|
|
21
|
+
value_ids = [value_ids]
|
|
22
|
+
|
|
23
|
+
if isinstance(value_ids, list):
|
|
24
|
+
for value_id in value_ids:
|
|
25
|
+
try:
|
|
26
|
+
yield {
|
|
27
|
+
"value": value_id,
|
|
28
|
+
"label": str(queryset.get(id=value_id)),
|
|
29
|
+
}
|
|
30
|
+
except ObjectDoesNotExist:
|
|
31
|
+
raise ParseError("Filter value invalid")
|
|
32
|
+
|
|
33
|
+
def get_representation(self, request, name, view):
|
|
34
|
+
representation, lookup_expr = super().get_representation(request, name, view)
|
|
35
|
+
lookup_expr["input_properties"]["multiple"] = self.MULTIPLE
|
|
36
|
+
|
|
37
|
+
queryset = self.get_queryset(request)
|
|
38
|
+
|
|
39
|
+
if hasattr(queryset.model, "get_label_key"):
|
|
40
|
+
label_key = queryset.model.get_label_key()
|
|
41
|
+
else:
|
|
42
|
+
label_key = self.label_key
|
|
43
|
+
|
|
44
|
+
url = reverse(self.endpoint, request=request)
|
|
45
|
+
if self.filter_params:
|
|
46
|
+
if callable(self.filter_params):
|
|
47
|
+
url += f"?{urlencode(self.filter_params(request, view))}"
|
|
48
|
+
else:
|
|
49
|
+
url += f"?{urlencode(self.filter_params)}"
|
|
50
|
+
lookup_expr["input_properties"]["endpoint"] = {
|
|
51
|
+
"url": url,
|
|
52
|
+
"value_key": self.value_key,
|
|
53
|
+
"label_key": label_key,
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
if default_ids := lookup_expr["input_properties"].get("default", None):
|
|
57
|
+
values = self.get_parsed_values(queryset, default_ids)
|
|
58
|
+
lookup_expr["input_properties"]["default"] = list(values) if self.MULTIPLE else next(values)
|
|
59
|
+
|
|
60
|
+
if initial_ids := lookup_expr["input_properties"].get("initial", None):
|
|
61
|
+
values = self.get_parsed_values(queryset, initial_ids)
|
|
62
|
+
lookup_expr["input_properties"]["initial"] = list(values) if self.MULTIPLE else next(values)
|
|
63
|
+
|
|
64
|
+
return representation, lookup_expr
|
|
65
|
+
|
|
66
|
+
|
|
67
|
+
class ModelMultipleChoiceFilter(ModelChoiceFilterMixin, django_filters.ModelMultipleChoiceFilter):
|
|
68
|
+
class SimpleModelMultipleChoiceField(ModelMultipleChoiceField):
|
|
69
|
+
"""
|
|
70
|
+
field class that define a simple text input as widget (instead of select). This is necessary in order to use the
|
|
71
|
+
browsable api for model with a lot of items. Without it, the widget would load all the queryset option and will probably destroy the performance
|
|
72
|
+
"""
|
|
73
|
+
|
|
74
|
+
class SimpleSelectMultiple(SelectMultiple):
|
|
75
|
+
template_name = "django/forms/widgets/text.html"
|
|
76
|
+
|
|
77
|
+
widget = SimpleSelectMultiple
|
|
78
|
+
|
|
79
|
+
MULTIPLE: bool = True
|
|
80
|
+
field_class = SimpleModelMultipleChoiceField
|
|
81
|
+
filter_type = "select"
|
|
82
|
+
|
|
83
|
+
def __init__(self, *args, **kwargs):
|
|
84
|
+
self.endpoint = kwargs.pop("endpoint", None)
|
|
85
|
+
self.filter_params = kwargs.pop("filter_params", None)
|
|
86
|
+
self.value_key = kwargs.pop("value_key", None)
|
|
87
|
+
self.label_key = kwargs.pop("label_key", None)
|
|
88
|
+
|
|
89
|
+
# TODO: This is monkeypatched. Make sure that the CSVWidget is set here and only here!
|
|
90
|
+
if "widget" not in kwargs:
|
|
91
|
+
kwargs["widget"] = django_filters.widgets.CSVWidget
|
|
92
|
+
super().__init__(*args, **kwargs)
|
|
93
|
+
|
|
94
|
+
|
|
95
|
+
class ModelChoiceFilter(ModelChoiceFilterMixin, django_filters.ModelChoiceFilter):
|
|
96
|
+
class SimpleModelChoiceField(ModelChoiceField):
|
|
97
|
+
"""
|
|
98
|
+
field class that define a simple text input as widget (instead of select). This is necessary in order to use the
|
|
99
|
+
browsable api for model with a lot of items. Without it, the widget would load all the queryset option and will probably destroy the performance
|
|
100
|
+
"""
|
|
101
|
+
|
|
102
|
+
widget = TextInput
|
|
103
|
+
|
|
104
|
+
field_class = SimpleModelChoiceField
|
|
105
|
+
filter_type = "select"
|
|
106
|
+
MULTIPLE: bool = False
|
|
107
|
+
|
|
108
|
+
def __init__(self, *args, **kwargs):
|
|
109
|
+
self.endpoint = kwargs.pop("endpoint", None)
|
|
110
|
+
self.value_key = kwargs.pop("value_key", None)
|
|
111
|
+
self.filter_params = kwargs.pop("filter_params", None)
|
|
112
|
+
self.label_key = kwargs.pop("label_key", None)
|
|
113
|
+
super().__init__(*args, **kwargs)
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
from django_filters import Filter
|
|
2
|
+
from wbcore.filters.mixins import WBCoreFilterMixin
|
|
3
|
+
|
|
4
|
+
|
|
5
|
+
class MultipleLookupFilter(WBCoreFilterMixin):
|
|
6
|
+
def __init__(self, field_class, lookup_expr, **kwargs):
|
|
7
|
+
self.field_class = field_class
|
|
8
|
+
self.lookup_expr = lookup_expr
|
|
9
|
+
self.kwargs = kwargs
|
|
10
|
+
|
|
11
|
+
def get_filters(self, field_name) -> dict[str, Filter]:
|
|
12
|
+
filters = dict()
|
|
13
|
+
for lookup_expr in self.lookup_expr:
|
|
14
|
+
filters[f"{field_name}__{lookup_expr}"] = self.field_class(
|
|
15
|
+
lookup_expr=lookup_expr,
|
|
16
|
+
**self.kwargs,
|
|
17
|
+
)
|
|
18
|
+
|
|
19
|
+
return filters
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
import django_filters
|
|
2
|
+
from wbcore.filters.mixins import WBCoreFilterMixin
|
|
3
|
+
|
|
4
|
+
|
|
5
|
+
class NumberFilter(WBCoreFilterMixin, django_filters.NumberFilter):
|
|
6
|
+
filter_type = "number"
|
|
7
|
+
|
|
8
|
+
def __init__(self, precision: int = 0, percent: bool = False, delimiter=",", decimal_mark=".", *args, **kwargs):
|
|
9
|
+
self.precision = precision
|
|
10
|
+
self.percent = percent
|
|
11
|
+
self.delimiter = delimiter
|
|
12
|
+
self.decimal_mark = decimal_mark
|
|
13
|
+
super().__init__(*args, **kwargs)
|
|
14
|
+
|
|
15
|
+
def get_representation(self, request, name, view):
|
|
16
|
+
representation, lookup_expr = super().get_representation(request, name, view)
|
|
17
|
+
representation["precision"] = self.precision
|
|
18
|
+
representation["delimiter"] = self.delimiter
|
|
19
|
+
representation["decimal_mark"] = self.decimal_mark
|
|
20
|
+
return representation, lookup_expr
|
|
21
|
+
|
|
22
|
+
def filter(self, qs, value):
|
|
23
|
+
if self.percent and value is not None:
|
|
24
|
+
value /= 100
|
|
25
|
+
return super().filter(qs, value)
|
|
26
|
+
|
|
27
|
+
|
|
28
|
+
class YearFilter(NumberFilter):
|
|
29
|
+
def __init__(self, *args, **kwargs):
|
|
30
|
+
super().__init__(*args, **kwargs)
|
|
31
|
+
self.delimiter = ""
|
|
32
|
+
self.decimal_mark = "."
|
|
33
|
+
|
|
34
|
+
|
|
35
|
+
class RangeSelectFilter(NumberFilter):
|
|
36
|
+
filter_type = "rangeselect"
|
|
37
|
+
|
|
38
|
+
def __init__(self, precision=2, *args, **kwargs):
|
|
39
|
+
self.precision = precision
|
|
40
|
+
self.color = kwargs.pop("color", "rgb(133, 144, 162)")
|
|
41
|
+
super().__init__(*args, **kwargs)
|
|
42
|
+
|
|
43
|
+
def get_representation(self, request, name, view):
|
|
44
|
+
representation, lookup_expr = super().get_representation(request, name, view)
|
|
45
|
+
lookup_expr["input_properties"]["color"] = self.color
|
|
46
|
+
return representation, lookup_expr
|
|
@@ -0,0 +1,217 @@
|
|
|
1
|
+
import logging
|
|
2
|
+
|
|
3
|
+
from django.contrib.postgres.fields import DateRangeField, DateTimeRangeField
|
|
4
|
+
from django.db import models
|
|
5
|
+
from django.db.models.fields.related import ManyToManyRel, ManyToOneRel, OneToOneRel
|
|
6
|
+
from django_filters.filterset import FilterSetMetaclass, remote_queryset, settings
|
|
7
|
+
from django_filters.rest_framework import FilterSet as DjangoFilterSet
|
|
8
|
+
from wbcore.filters import fields
|
|
9
|
+
from wbcore.filters.fields.multiple_lookups import MultipleLookupFilter
|
|
10
|
+
from wbcore.signals.filters import add_filters
|
|
11
|
+
|
|
12
|
+
from .utils import check_required_filters
|
|
13
|
+
|
|
14
|
+
logger = logging.getLogger(__name__)
|
|
15
|
+
|
|
16
|
+
|
|
17
|
+
def _is_number(field):
|
|
18
|
+
return (
|
|
19
|
+
issubclass(field.__class__, models.DecimalField)
|
|
20
|
+
or issubclass(field.__class__, models.FloatField)
|
|
21
|
+
or issubclass(field.__class__, models.IntegerField)
|
|
22
|
+
)
|
|
23
|
+
|
|
24
|
+
|
|
25
|
+
class CustomFilterSetMetaClass(FilterSetMetaclass):
|
|
26
|
+
def __new__(cls, name, bases, attrs):
|
|
27
|
+
new_class = super().__new__(cls, name, bases, attrs)
|
|
28
|
+
if _meta := getattr(new_class, "Meta", None):
|
|
29
|
+
for parent_field_name, child_fields in getattr(_meta, "flatten_fields", dict()).items():
|
|
30
|
+
if remote_field := getattr(_meta.model._meta.get_field(parent_field_name), "remote_field", None):
|
|
31
|
+
for child_field_name, lookup_exps in child_fields.items():
|
|
32
|
+
child_field = remote_field.model._meta.get_field(child_field_name)
|
|
33
|
+
if _is_number(child_field):
|
|
34
|
+
nested_field_name = f"{parent_field_name}__{child_field_name}"
|
|
35
|
+
for lookup_exp in lookup_exps:
|
|
36
|
+
new_class.base_filters[f"{nested_field_name}__{lookup_exp}"] = fields.NumberFilter(
|
|
37
|
+
field_name=nested_field_name,
|
|
38
|
+
lookup_expr=lookup_exp,
|
|
39
|
+
precision=2,
|
|
40
|
+
label=child_field.verbose_name,
|
|
41
|
+
)
|
|
42
|
+
|
|
43
|
+
# TODO extend with other filter fields type
|
|
44
|
+
return new_class
|
|
45
|
+
|
|
46
|
+
@classmethod
|
|
47
|
+
def get_declared_filters(cls, bases, attrs):
|
|
48
|
+
filters = super().get_declared_filters(bases, attrs)
|
|
49
|
+
|
|
50
|
+
multi_filters: list[tuple[str, MultipleLookupFilter]] = [
|
|
51
|
+
(filter_name, attrs.pop(filter_name))
|
|
52
|
+
for filter_name, obj in list(attrs.items())
|
|
53
|
+
if isinstance(obj, MultipleLookupFilter)
|
|
54
|
+
]
|
|
55
|
+
|
|
56
|
+
for field_name, filter in multi_filters:
|
|
57
|
+
filters |= filter.get_filters(field_name)
|
|
58
|
+
|
|
59
|
+
return filters
|
|
60
|
+
|
|
61
|
+
|
|
62
|
+
class FilterSet(DjangoFilterSet, metaclass=CustomFilterSetMetaClass):
|
|
63
|
+
FILTER_DEFAULTS = {
|
|
64
|
+
models.BooleanField: {"filter_class": fields.BooleanFilter},
|
|
65
|
+
models.NullBooleanField: {"filter_class": fields.BooleanFilter},
|
|
66
|
+
models.CharField: {"filter_class": fields.CharFilter},
|
|
67
|
+
models.TextField: {"filter_class": fields.CharFilter},
|
|
68
|
+
models.SlugField: {"filter_class": fields.CharFilter},
|
|
69
|
+
models.EmailField: {"filter_class": fields.CharFilter},
|
|
70
|
+
models.FilePathField: {"filter_class": fields.CharFilter},
|
|
71
|
+
models.UUIDField: {"filter_class": fields.CharFilter},
|
|
72
|
+
models.URLField: {"filter_class": fields.CharFilter},
|
|
73
|
+
models.GenericIPAddressField: {"filter_class": fields.CharFilter},
|
|
74
|
+
models.CommaSeparatedIntegerField: {"filter_class": fields.CharFilter},
|
|
75
|
+
models.DateField: {"filter_class": fields.DateFilter},
|
|
76
|
+
models.DateTimeField: {"filter_class": fields.DateTimeFilter},
|
|
77
|
+
models.TimeField: {"filter_class": fields.TimeFilter},
|
|
78
|
+
models.IntegerField: {"filter_class": fields.NumberFilter},
|
|
79
|
+
models.FloatField: {"filter_class": fields.NumberFilter},
|
|
80
|
+
models.DecimalField: {"filter_class": fields.NumberFilter},
|
|
81
|
+
DateTimeRangeField: {"filter_class": fields.DateTimeRangeFilter},
|
|
82
|
+
DateRangeField: {"filter_class": fields.DateRangeFilter},
|
|
83
|
+
# models.DurationField: {"filter_class": DurationFilter},
|
|
84
|
+
# models.SmallIntegerField: {"filter_class": NumberFilter},
|
|
85
|
+
# models.AutoField: {"filter_class": NumberFilter},
|
|
86
|
+
# models.PositiveIntegerField: {"filter_class": NumberFilter},
|
|
87
|
+
# models.PositiveSmallIntegerField: {"filter_class": NumberFilter},
|
|
88
|
+
# models.UUIDField: {"filter_class": UUIDFilter},
|
|
89
|
+
# Forward relationships
|
|
90
|
+
models.OneToOneField: {
|
|
91
|
+
"filter_class": fields.ModelChoiceFilter,
|
|
92
|
+
"extra": lambda f: {
|
|
93
|
+
"queryset": remote_queryset(f),
|
|
94
|
+
"endpoint": f.related_model.get_representation_endpoint(),
|
|
95
|
+
"value_key": f.related_model.get_representation_value_key(),
|
|
96
|
+
"label_key": f.related_model.get_representation_label_key(),
|
|
97
|
+
"to_field_name": f.remote_field.field_name,
|
|
98
|
+
"null_label": settings.NULL_CHOICE_LABEL if f.null else None,
|
|
99
|
+
},
|
|
100
|
+
},
|
|
101
|
+
models.ForeignKey: {
|
|
102
|
+
"filter_class": fields.ModelChoiceFilter,
|
|
103
|
+
"extra": lambda f: {
|
|
104
|
+
"queryset": remote_queryset(f),
|
|
105
|
+
"endpoint": f.related_model.get_representation_endpoint(),
|
|
106
|
+
"value_key": f.related_model.get_representation_value_key(),
|
|
107
|
+
"label_key": f.related_model.get_representation_label_key(),
|
|
108
|
+
"to_field_name": f.remote_field.field_name,
|
|
109
|
+
"null_label": settings.NULL_CHOICE_LABEL if f.null else None,
|
|
110
|
+
},
|
|
111
|
+
},
|
|
112
|
+
models.ManyToManyField: {
|
|
113
|
+
"filter_class": fields.ModelMultipleChoiceFilter,
|
|
114
|
+
"extra": lambda f: {
|
|
115
|
+
"queryset": remote_queryset(f),
|
|
116
|
+
"endpoint": f.related_model.get_representation_endpoint(),
|
|
117
|
+
"value_key": f.related_model.get_representation_value_key(),
|
|
118
|
+
"label_key": f.related_model.get_representation_label_key(),
|
|
119
|
+
},
|
|
120
|
+
},
|
|
121
|
+
# Reverse relationships
|
|
122
|
+
OneToOneRel: {
|
|
123
|
+
"filter_class": fields.ModelChoiceFilter,
|
|
124
|
+
"extra": lambda f: {
|
|
125
|
+
"queryset": remote_queryset(f),
|
|
126
|
+
"endpoint": f.related_model.get_representation_endpoint(),
|
|
127
|
+
"value_key": f.related_model.get_representation_value_key(),
|
|
128
|
+
"label_key": f.related_model.get_representation_label_key(),
|
|
129
|
+
"null_label": settings.NULL_CHOICE_LABEL if f.null else None,
|
|
130
|
+
},
|
|
131
|
+
},
|
|
132
|
+
ManyToOneRel: {
|
|
133
|
+
"filter_class": fields.ModelMultipleChoiceFilter,
|
|
134
|
+
"extra": lambda f: {
|
|
135
|
+
"queryset": remote_queryset(f),
|
|
136
|
+
"endpoint": f.related_model.get_representation_endpoint(),
|
|
137
|
+
"value_key": f.related_model.get_representation_value_key(),
|
|
138
|
+
"label_key": f.related_model.get_representation_label_key(),
|
|
139
|
+
},
|
|
140
|
+
},
|
|
141
|
+
ManyToManyRel: {
|
|
142
|
+
"filter_class": fields.ModelMultipleChoiceFilter,
|
|
143
|
+
"extra": lambda f: {
|
|
144
|
+
"queryset": remote_queryset(f),
|
|
145
|
+
"endpoint": f.related_model.get_representation_endpoint(),
|
|
146
|
+
"value_key": f.related_model.get_representation_value_key(),
|
|
147
|
+
"label_key": f.related_model.get_representation_label_key(),
|
|
148
|
+
},
|
|
149
|
+
},
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
@classmethod
|
|
153
|
+
def filter_class_for_remote_filter(cls):
|
|
154
|
+
if hasattr(cls, "get_filter_class_for_remote_filter") and callable(cls.get_filter_class_for_remote_filter()):
|
|
155
|
+
return cls.get_filter_class_for_remote_filter()
|
|
156
|
+
return cls
|
|
157
|
+
|
|
158
|
+
def __init__(self, data=None, queryset=None, *, request=None, prefix=None, view=None):
|
|
159
|
+
super().__init__(data=data, queryset=queryset, request=request, prefix=prefix)
|
|
160
|
+
remote_filters = add_filters.send(sender=self.filter_class_for_remote_filter(), request=request)
|
|
161
|
+
self.view = view
|
|
162
|
+
for _, res in remote_filters:
|
|
163
|
+
if res:
|
|
164
|
+
for remote_filter_key, remote_filter in res.items():
|
|
165
|
+
setattr(remote_filter, "column_field_name", remote_filter_key)
|
|
166
|
+
self.filters[remote_filter_key] = remote_filter
|
|
167
|
+
|
|
168
|
+
@classmethod
|
|
169
|
+
def filter_for_lookup(cls, field, lookup_type):
|
|
170
|
+
if lookup_type == "exact" and getattr(field, "choices", None):
|
|
171
|
+
filter_class, params = fields.ChoiceFilter, {"choices": field.choices}
|
|
172
|
+
else:
|
|
173
|
+
filter_class, params = super().filter_for_lookup(field, lookup_type)
|
|
174
|
+
|
|
175
|
+
# Check if it is a decimal field:
|
|
176
|
+
if hasattr(field, "decimal_places"):
|
|
177
|
+
params["precision"] = field.decimal_places
|
|
178
|
+
|
|
179
|
+
if hasattr(field, "verbose_name"):
|
|
180
|
+
params["label"] = field.verbose_name
|
|
181
|
+
|
|
182
|
+
return filter_class, params
|
|
183
|
+
|
|
184
|
+
@classmethod
|
|
185
|
+
def get_dependency_map(cls):
|
|
186
|
+
if hasattr(cls, "Meta"):
|
|
187
|
+
yield from getattr(cls.Meta, "dependency_map", dict()).items()
|
|
188
|
+
|
|
189
|
+
@classmethod
|
|
190
|
+
def get_filters(cls):
|
|
191
|
+
filters = super().get_filters()
|
|
192
|
+
remote_filters = add_filters.send(sender=cls.filter_class_for_remote_filter())
|
|
193
|
+
for _, res in remote_filters:
|
|
194
|
+
if res:
|
|
195
|
+
for remote_filter_key, remote_filter in res.items():
|
|
196
|
+
setattr(remote_filter, "column_field_name", remote_filter_key)
|
|
197
|
+
filters[remote_filter_key] = remote_filter
|
|
198
|
+
|
|
199
|
+
for field, help_text in getattr(cls, "help_texts", {}).items():
|
|
200
|
+
filters[field].help_text = help_text
|
|
201
|
+
|
|
202
|
+
# Get dependency map specified in Meta and append the argument accordingly
|
|
203
|
+
for field, values in cls.get_dependency_map():
|
|
204
|
+
for value in values:
|
|
205
|
+
filters[field].depends_on.append({"field": value, "options": {}})
|
|
206
|
+
return filters
|
|
207
|
+
|
|
208
|
+
def extract_required_field_labels(self):
|
|
209
|
+
return [label for label, filter in self.base_filters.items() if getattr(filter, "required", False)]
|
|
210
|
+
|
|
211
|
+
def filter_queryset(self, queryset):
|
|
212
|
+
queryset = super().filter_queryset(queryset)
|
|
213
|
+
check_required_filters(self.request, self.view, self)
|
|
214
|
+
return queryset
|
|
215
|
+
|
|
216
|
+
def fake_filter(self, queryset, name, value):
|
|
217
|
+
return queryset
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
LOOKUP_EXPRESSION_MAPPING = {
|
|
2
|
+
"exact": ("Equals", "="),
|
|
3
|
+
"iexact": ("Equals (case-insensitive)", "≈"),
|
|
4
|
+
"contains": ("Contains", "[=]"),
|
|
5
|
+
"icontains": ("Contains (case-insensitive)", "[≈]"),
|
|
6
|
+
"startswith": ("Starts with", "=]"),
|
|
7
|
+
"istartswith": ("Starts with (case-insensitive)", "≈]"),
|
|
8
|
+
"endswith": ("Ends with", "[="),
|
|
9
|
+
"iendswith": ("Ends with (case-insensitive)", "[≈"),
|
|
10
|
+
"regex": ("Regex", "r="),
|
|
11
|
+
"iregex": ("Regex (case-insensitive)", "r≈"),
|
|
12
|
+
"in": ("In", "∈"),
|
|
13
|
+
"gte": ("Greater than or Equal", ">="),
|
|
14
|
+
"lte": ("Less than or Equal", "<="),
|
|
15
|
+
"gt": ("Greater than", ">"),
|
|
16
|
+
"lt": ("Less than", "<"),
|
|
17
|
+
"overlap": ("Overlaps", "="),
|
|
18
|
+
"isnull": ("Is Null", "Is Null"),
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
|
|
22
|
+
ALL_TEXT_LOOKUPS = [
|
|
23
|
+
"exact",
|
|
24
|
+
"iexact",
|
|
25
|
+
"contains",
|
|
26
|
+
"icontains",
|
|
27
|
+
"startswith",
|
|
28
|
+
"istartswith",
|
|
29
|
+
"endswith",
|
|
30
|
+
"iendswith",
|
|
31
|
+
"regex",
|
|
32
|
+
"iregex",
|
|
33
|
+
]
|
|
34
|
+
|
|
35
|
+
|
|
36
|
+
def get_lookup_label(lookup_expr: str) -> str:
|
|
37
|
+
return LOOKUP_EXPRESSION_MAPPING.get(lookup_expr, (lookup_expr, lookup_expr))[0]
|
|
38
|
+
|
|
39
|
+
|
|
40
|
+
def get_lookup_icon(lookup_expr: str) -> str:
|
|
41
|
+
return LOOKUP_EXPRESSION_MAPPING.get(lookup_expr, (lookup_expr, lookup_expr))[1]
|
wbcore/filters/mixins.py
ADDED
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
from contextlib import suppress
|
|
2
|
+
|
|
3
|
+
from django_filters.utils import get_model_field
|
|
4
|
+
|
|
5
|
+
from .lookups import get_lookup_icon, get_lookup_label
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
class WBCoreFilterMixin:
|
|
9
|
+
def __init__(self, *args, **kwargs):
|
|
10
|
+
self.default = kwargs.pop("default", None)
|
|
11
|
+
self.initial = kwargs.pop("initial", None)
|
|
12
|
+
self.required = kwargs.pop("required", False)
|
|
13
|
+
self.clearable = kwargs.pop("clearable", True) # TODO: Take away
|
|
14
|
+
self.hidden = kwargs.pop("hidden", False)
|
|
15
|
+
self.column_field_name = kwargs.pop("column_field_name", None)
|
|
16
|
+
self.help_text = kwargs.pop("help_text", None)
|
|
17
|
+
self.allow_empty_default = kwargs.pop("allow_empty_default", False)
|
|
18
|
+
self.label_format = kwargs.pop(
|
|
19
|
+
"label_format",
|
|
20
|
+
getattr(self, "default_label_format", "{{field_label}} {{operation_icon}} {{value_label}}"),
|
|
21
|
+
)
|
|
22
|
+
self.lookup_icon = kwargs.pop("lookup_icon", None)
|
|
23
|
+
self.lookup_label = kwargs.pop("lookup_label", None)
|
|
24
|
+
self.depends_on = kwargs.pop("depends_on", [])
|
|
25
|
+
super().__init__(*args, **kwargs)
|
|
26
|
+
|
|
27
|
+
@property
|
|
28
|
+
def key(self):
|
|
29
|
+
return self.column_field_name if self.column_field_name else self.field_name
|
|
30
|
+
|
|
31
|
+
def get_label(self):
|
|
32
|
+
if self.label is None: # if label is not provided we gracefully convert the field name into capitalized label
|
|
33
|
+
return self.field_name.replace("_", " ").title()
|
|
34
|
+
else:
|
|
35
|
+
return self.label
|
|
36
|
+
|
|
37
|
+
def _get_default(self, request, view):
|
|
38
|
+
# We consider the case where default is a boolean with value False.
|
|
39
|
+
if callable(self.default):
|
|
40
|
+
default = self.default(self, request, view)
|
|
41
|
+
elif (
|
|
42
|
+
isinstance(self.default, str)
|
|
43
|
+
and (callable_default := getattr(self, self.default, None))
|
|
44
|
+
and (callable(callable_default))
|
|
45
|
+
):
|
|
46
|
+
default = callable_default(self, request, view)
|
|
47
|
+
else:
|
|
48
|
+
default = self.default
|
|
49
|
+
|
|
50
|
+
return default
|
|
51
|
+
|
|
52
|
+
def _validate_default_with_request(self, default, request, name):
|
|
53
|
+
if request_default := request.GET.get(name):
|
|
54
|
+
return request_default
|
|
55
|
+
return default
|
|
56
|
+
|
|
57
|
+
def get_help_text(self) -> str:
|
|
58
|
+
if self.help_text:
|
|
59
|
+
return self.help_text
|
|
60
|
+
with suppress(AttributeError):
|
|
61
|
+
field = get_model_field(self.parent._meta.model, self.field_name)
|
|
62
|
+
if field.help_text:
|
|
63
|
+
return field.help_text
|
|
64
|
+
if self.label:
|
|
65
|
+
return "Filter by " + self.label
|
|
66
|
+
|
|
67
|
+
def get_representation(self, request, name, view):
|
|
68
|
+
if self.hidden:
|
|
69
|
+
return {}
|
|
70
|
+
|
|
71
|
+
representation = {
|
|
72
|
+
"key": self.key,
|
|
73
|
+
"label_format": self.label_format,
|
|
74
|
+
"label": self.get_label(),
|
|
75
|
+
"help_text": self.get_help_text(),
|
|
76
|
+
}
|
|
77
|
+
lookup_expr = {
|
|
78
|
+
"label": get_lookup_label(self.lookup_expr) if self.lookup_label is None else self.lookup_label,
|
|
79
|
+
"icon": get_lookup_icon(self.lookup_expr) if self.lookup_icon is None else self.lookup_icon,
|
|
80
|
+
"key": name,
|
|
81
|
+
"hidden": self.hidden,
|
|
82
|
+
"input_properties": {
|
|
83
|
+
"type": self.filter_type,
|
|
84
|
+
},
|
|
85
|
+
}
|
|
86
|
+
default = self._get_default(request, view)
|
|
87
|
+
if default is not None or self.allow_empty_default:
|
|
88
|
+
lookup_expr["input_properties"]["default"] = default
|
|
89
|
+
|
|
90
|
+
initial = None
|
|
91
|
+
if callable(self.initial):
|
|
92
|
+
initial = self.initial(self, request, view)
|
|
93
|
+
elif (
|
|
94
|
+
isinstance(self.initial, str)
|
|
95
|
+
and (callable_initial := getattr(self, self.initial, None))
|
|
96
|
+
and (callable(callable_initial))
|
|
97
|
+
):
|
|
98
|
+
initial = callable_initial(self, request, view)
|
|
99
|
+
else:
|
|
100
|
+
initial = self.initial
|
|
101
|
+
|
|
102
|
+
if _initial := self._validate_default_with_request(initial, request, name):
|
|
103
|
+
initial = _initial
|
|
104
|
+
|
|
105
|
+
if initial is not None:
|
|
106
|
+
lookup_expr["input_properties"]["initial"] = initial
|
|
107
|
+
|
|
108
|
+
if self.required:
|
|
109
|
+
lookup_expr["input_properties"]["required"] = True
|
|
110
|
+
# assert representation["default"] != {}, "If a filter is required, it needs at least one default value"
|
|
111
|
+
representation["depends_on"] = self.depends_on
|
|
112
|
+
return representation, lookup_expr
|
wbcore/filters/utils.py
ADDED
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import os
|
|
2
|
+
|
|
3
|
+
from django_filters.constants import EMPTY_VALUES
|
|
4
|
+
|
|
5
|
+
from .defaults import RequiredFilterMissing
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
def check_required_filters(request, view, filter):
|
|
9
|
+
if "PYTEST_CURRENT_TEST" not in os.environ and getattr(view, "action", "list") == "list":
|
|
10
|
+
errors = []
|
|
11
|
+
for required_field in [
|
|
12
|
+
label for label, filter in filter.base_filters.items() if getattr(filter, "required", False)
|
|
13
|
+
]:
|
|
14
|
+
if filter.form.cleaned_data.get(required_field, None) in EMPTY_VALUES:
|
|
15
|
+
errors.append(required_field)
|
|
16
|
+
if errors:
|
|
17
|
+
raise RequiredFilterMissing(
|
|
18
|
+
detail={
|
|
19
|
+
"filter_errors": {field: ["This filter is required. Please specify a value"] for field in errors},
|
|
20
|
+
}
|
|
21
|
+
)
|