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.
Files changed (1035) hide show
  1. wbcore/__init__.py +1 -0
  2. wbcore/admin.py +197 -0
  3. wbcore/apps.py +17 -0
  4. wbcore/cache/__init__.py +0 -0
  5. wbcore/cache/buttons.py +22 -0
  6. wbcore/cache/decorators.py +29 -0
  7. wbcore/cache/mixins.py +48 -0
  8. wbcore/cache/registry.py +90 -0
  9. wbcore/cache/views.py +18 -0
  10. wbcore/configs/__init__.py +11 -0
  11. wbcore/configs/configs.py +51 -0
  12. wbcore/configs/decorators.py +11 -0
  13. wbcore/configs/registry.py +35 -0
  14. wbcore/configs/views.py +11 -0
  15. wbcore/configurations/__init__.py +1 -0
  16. wbcore/configurations/base.py +45 -0
  17. wbcore/configurations/configurations/__init__.py +16 -0
  18. wbcore/configurations/configurations/apps.py +54 -0
  19. wbcore/configurations/configurations/authentication.py +43 -0
  20. wbcore/configurations/configurations/base.py +15 -0
  21. wbcore/configurations/configurations/cache.py +20 -0
  22. wbcore/configurations/configurations/celery.py +26 -0
  23. wbcore/configurations/configurations/i18nl10n.py +10 -0
  24. wbcore/configurations/configurations/mail.py +2 -0
  25. wbcore/configurations/configurations/maintenance.py +53 -0
  26. wbcore/configurations/configurations/media.py +22 -0
  27. wbcore/configurations/configurations/middleware.py +27 -0
  28. wbcore/configurations/configurations/network.py +19 -0
  29. wbcore/configurations/configurations/rest_framework.py +42 -0
  30. wbcore/configurations/configurations/static.py +28 -0
  31. wbcore/configurations/configurations/templates.py +17 -0
  32. wbcore/configurations/configurations/uvicorn.py +9 -0
  33. wbcore/configurations/configurations/wbcore.py +68 -0
  34. wbcore/content_type/__init__.py +0 -0
  35. wbcore/content_type/admin.py +8 -0
  36. wbcore/content_type/filters.py +19 -0
  37. wbcore/content_type/serializers.py +89 -0
  38. wbcore/content_type/utils.py +30 -0
  39. wbcore/content_type/viewsets.py +80 -0
  40. wbcore/contrib/__init__.py +0 -0
  41. wbcore/contrib/agenda/__init__.py +0 -0
  42. wbcore/contrib/agenda/admin/__init__.py +2 -0
  43. wbcore/contrib/agenda/admin/calendar_item.py +13 -0
  44. wbcore/contrib/agenda/admin/conference_room.py +18 -0
  45. wbcore/contrib/agenda/apps.py +6 -0
  46. wbcore/contrib/agenda/configurations.py +11 -0
  47. wbcore/contrib/agenda/factories/__init__.py +2 -0
  48. wbcore/contrib/agenda/factories/calendar_item.py +47 -0
  49. wbcore/contrib/agenda/factories/conference_room.py +19 -0
  50. wbcore/contrib/agenda/filters/__init__.py +2 -0
  51. wbcore/contrib/agenda/filters/calendar_item.py +64 -0
  52. wbcore/contrib/agenda/filters/conference_room.py +41 -0
  53. wbcore/contrib/agenda/migrations/0001_initial.py +84 -0
  54. wbcore/contrib/agenda/migrations/0002_initial.py +26 -0
  55. wbcore/contrib/agenda/migrations/0003_calendaritem_endpoint_basename.py +42 -0
  56. wbcore/contrib/agenda/migrations/0004_alter_calendaritem_item_type.py +17 -0
  57. wbcore/contrib/agenda/migrations/0005_building_and_more.py +94 -0
  58. wbcore/contrib/agenda/migrations/0006_calendaritem_is_deletable.py +17 -0
  59. wbcore/contrib/agenda/migrations/0007_alter_calendaritem_options.py +21 -0
  60. wbcore/contrib/agenda/migrations/0008_alter_calendaritem_item_type.py +17 -0
  61. wbcore/contrib/agenda/migrations/0009_alter_calendaritem_icon.py +18 -0
  62. wbcore/contrib/agenda/migrations/__init__.py +0 -0
  63. wbcore/contrib/agenda/models/__init__.py +2 -0
  64. wbcore/contrib/agenda/models/calendar_item.py +238 -0
  65. wbcore/contrib/agenda/models/conference_room.py +95 -0
  66. wbcore/contrib/agenda/release_notes/1_0_0.md +13 -0
  67. wbcore/contrib/agenda/release_notes/__init__.py +0 -0
  68. wbcore/contrib/agenda/serializers/__init__.py +10 -0
  69. wbcore/contrib/agenda/serializers/calendar_item.py +76 -0
  70. wbcore/contrib/agenda/serializers/conference_room.py +98 -0
  71. wbcore/contrib/agenda/signals.py +8 -0
  72. wbcore/contrib/agenda/tests/__init__.py +0 -0
  73. wbcore/contrib/agenda/tests/conftest.py +14 -0
  74. wbcore/contrib/agenda/tests/signals.py +16 -0
  75. wbcore/contrib/agenda/tests/test_models.py +34 -0
  76. wbcore/contrib/agenda/tests/test_viewsets.py +171 -0
  77. wbcore/contrib/agenda/tests/tests.py +25 -0
  78. wbcore/contrib/agenda/typings.py +19 -0
  79. wbcore/contrib/agenda/urls.py +25 -0
  80. wbcore/contrib/agenda/viewsets/__init__.py +13 -0
  81. wbcore/contrib/agenda/viewsets/buttons/__init__.py +1 -0
  82. wbcore/contrib/agenda/viewsets/buttons/conference_room.py +20 -0
  83. wbcore/contrib/agenda/viewsets/calendar_items.py +168 -0
  84. wbcore/contrib/agenda/viewsets/conference_room.py +48 -0
  85. wbcore/contrib/agenda/viewsets/display/__init__.py +2 -0
  86. wbcore/contrib/agenda/viewsets/display/calendar_items.py +17 -0
  87. wbcore/contrib/agenda/viewsets/display/conference_room.py +41 -0
  88. wbcore/contrib/agenda/viewsets/endpoints/__init__.py +1 -0
  89. wbcore/contrib/agenda/viewsets/endpoints/calendar_items.py +18 -0
  90. wbcore/contrib/agenda/viewsets/menu/__init__.py +2 -0
  91. wbcore/contrib/agenda/viewsets/menu/calendar_items.py +17 -0
  92. wbcore/contrib/agenda/viewsets/menu/conference_room.py +37 -0
  93. wbcore/contrib/agenda/viewsets/titles/__init__.py +2 -0
  94. wbcore/contrib/agenda/viewsets/titles/calendar_items.py +7 -0
  95. wbcore/contrib/agenda/viewsets/titles/conference_room.py +24 -0
  96. wbcore/contrib/ai/__init__.py +0 -0
  97. wbcore/contrib/ai/apps.py +5 -0
  98. wbcore/contrib/ai/exceptions.py +42 -0
  99. wbcore/contrib/ai/llm/__init__.py +0 -0
  100. wbcore/contrib/ai/llm/config.py +107 -0
  101. wbcore/contrib/ai/llm/decorators.py +10 -0
  102. wbcore/contrib/ai/llm/mixins.py +35 -0
  103. wbcore/contrib/ai/llm/utils.py +45 -0
  104. wbcore/contrib/authentication/__init__.py +9 -0
  105. wbcore/contrib/authentication/admin.py +247 -0
  106. wbcore/contrib/authentication/apps.py +14 -0
  107. wbcore/contrib/authentication/authentication.py +162 -0
  108. wbcore/contrib/authentication/configs.py +9 -0
  109. wbcore/contrib/authentication/configurations.py +53 -0
  110. wbcore/contrib/authentication/dynamic_preferences_registry.py +26 -0
  111. wbcore/contrib/authentication/factories/__init__.py +3 -0
  112. wbcore/contrib/authentication/factories/tokens.py +15 -0
  113. wbcore/contrib/authentication/factories/users.py +82 -0
  114. wbcore/contrib/authentication/factories/users_activities.py +19 -0
  115. wbcore/contrib/authentication/filters.py +18 -0
  116. wbcore/contrib/authentication/management/__init__.py +14 -0
  117. wbcore/contrib/authentication/migrations/0001_initial_squashed.py +174 -0
  118. wbcore/contrib/authentication/migrations/0002_profile.py +26 -0
  119. wbcore/contrib/authentication/migrations/0003_alter_user_profile.py +34 -0
  120. wbcore/contrib/authentication/migrations/0004_token.py +51 -0
  121. wbcore/contrib/authentication/migrations/0005_user_external_calendar_settings.py +17 -0
  122. wbcore/contrib/authentication/migrations/0006_auto_20231206_1422.py +13 -0
  123. wbcore/contrib/authentication/migrations/__init__.py +0 -0
  124. wbcore/contrib/authentication/models/__init__.py +3 -0
  125. wbcore/contrib/authentication/models/tokens.py +140 -0
  126. wbcore/contrib/authentication/models/users.py +225 -0
  127. wbcore/contrib/authentication/models/users_activities.py +112 -0
  128. wbcore/contrib/authentication/release_notes/1_0_0.md +13 -0
  129. wbcore/contrib/authentication/release_notes/__init__.py +0 -0
  130. wbcore/contrib/authentication/serializers/__init__.py +14 -0
  131. wbcore/contrib/authentication/serializers/users.py +350 -0
  132. wbcore/contrib/authentication/serializers/users_activites.py +37 -0
  133. wbcore/contrib/authentication/tasks.py +29 -0
  134. wbcore/contrib/authentication/tests/__init__.py +0 -0
  135. wbcore/contrib/authentication/tests/conftest.py +18 -0
  136. wbcore/contrib/authentication/tests/e2e/__init__.py +1 -0
  137. wbcore/contrib/authentication/tests/e2e/e2e_auth_utility.py +20 -0
  138. wbcore/contrib/authentication/tests/signals.py +17 -0
  139. wbcore/contrib/authentication/tests/test_configs.py +6 -0
  140. wbcore/contrib/authentication/tests/test_serializers.py +6 -0
  141. wbcore/contrib/authentication/tests/test_tasks.py +33 -0
  142. wbcore/contrib/authentication/tests/test_tokens.py +126 -0
  143. wbcore/contrib/authentication/tests/test_users.py +338 -0
  144. wbcore/contrib/authentication/tests/test_viewsets.py +6 -0
  145. wbcore/contrib/authentication/tests/tests.py +15 -0
  146. wbcore/contrib/authentication/urls.py +97 -0
  147. wbcore/contrib/authentication/utils.py +10 -0
  148. wbcore/contrib/authentication/viewsets/__init__.py +19 -0
  149. wbcore/contrib/authentication/viewsets/buttons/__init__.py +1 -0
  150. wbcore/contrib/authentication/viewsets/buttons/users.py +55 -0
  151. wbcore/contrib/authentication/viewsets/display/__init__.py +6 -0
  152. wbcore/contrib/authentication/viewsets/display/user_activities.py +85 -0
  153. wbcore/contrib/authentication/viewsets/display/users.py +72 -0
  154. wbcore/contrib/authentication/viewsets/endpoints/__init__.py +6 -0
  155. wbcore/contrib/authentication/viewsets/endpoints/user_activities.py +22 -0
  156. wbcore/contrib/authentication/viewsets/endpoints/users.py +20 -0
  157. wbcore/contrib/authentication/viewsets/menu/__init__.py +6 -0
  158. wbcore/contrib/authentication/viewsets/menu/user_activities.py +39 -0
  159. wbcore/contrib/authentication/viewsets/menu/users.py +16 -0
  160. wbcore/contrib/authentication/viewsets/titles/__init__.py +12 -0
  161. wbcore/contrib/authentication/viewsets/titles/user_activities.py +31 -0
  162. wbcore/contrib/authentication/viewsets/titles/users.py +26 -0
  163. wbcore/contrib/authentication/viewsets/user_activities.py +222 -0
  164. wbcore/contrib/authentication/viewsets/users.py +328 -0
  165. wbcore/contrib/color/migrations/0001_initial.py +33 -0
  166. wbcore/contrib/color/migrations/0002_alter_colorgradient_colors.py +25 -0
  167. wbcore/contrib/color/migrations/__init__.py +0 -0
  168. wbcore/contrib/currency/__init__.py +0 -0
  169. wbcore/contrib/currency/admin.py +32 -0
  170. wbcore/contrib/currency/apps.py +5 -0
  171. wbcore/contrib/currency/dynamic_preferences_registry.py +39 -0
  172. wbcore/contrib/currency/factories.py +39 -0
  173. wbcore/contrib/currency/import_export/__init__.py +0 -0
  174. wbcore/contrib/currency/import_export/backends/__init__.py +1 -0
  175. wbcore/contrib/currency/import_export/backends/fixerio/__init__.py +1 -0
  176. wbcore/contrib/currency/import_export/backends/fixerio/currency_fx_rates.py +66 -0
  177. wbcore/contrib/currency/import_export/backends/utils.py +5 -0
  178. wbcore/contrib/currency/import_export/handlers/__init__.py +2 -0
  179. wbcore/contrib/currency/import_export/handlers/currency.py +24 -0
  180. wbcore/contrib/currency/import_export/handlers/currency_fx_rates.py +28 -0
  181. wbcore/contrib/currency/import_export/parsers/__init__.py +0 -0
  182. wbcore/contrib/currency/import_export/parsers/fixerio/__init__.py +0 -0
  183. wbcore/contrib/currency/import_export/parsers/fixerio/currency_fx_rates.py +34 -0
  184. wbcore/contrib/currency/migrations/0001_initial.py +89 -0
  185. wbcore/contrib/currency/migrations/__init__.py +0 -0
  186. wbcore/contrib/currency/models.py +176 -0
  187. wbcore/contrib/currency/release_notes/1_0_0.md +13 -0
  188. wbcore/contrib/currency/release_notes/__init__.py +0 -0
  189. wbcore/contrib/currency/serializers.py +40 -0
  190. wbcore/contrib/currency/tests/__init__.py +0 -0
  191. wbcore/contrib/currency/tests/conftest.py +7 -0
  192. wbcore/contrib/currency/tests/test_models.py +104 -0
  193. wbcore/contrib/currency/tests/test_serializers.py +62 -0
  194. wbcore/contrib/currency/tests/test_viewsets.py +197 -0
  195. wbcore/contrib/currency/urls.py +19 -0
  196. wbcore/contrib/currency/viewsets/__init__.py +2 -0
  197. wbcore/contrib/currency/viewsets/buttons/__init__.py +0 -0
  198. wbcore/contrib/currency/viewsets/currency.py +56 -0
  199. wbcore/contrib/currency/viewsets/currency_fx_rates.py +32 -0
  200. wbcore/contrib/currency/viewsets/display/__init__.py +2 -0
  201. wbcore/contrib/currency/viewsets/display/currency.py +37 -0
  202. wbcore/contrib/currency/viewsets/display/currency_fx_rates.py +9 -0
  203. wbcore/contrib/currency/viewsets/endpoints/__init__.py +1 -0
  204. wbcore/contrib/currency/viewsets/endpoints/currency_fx_rates.py +14 -0
  205. wbcore/contrib/currency/viewsets/menu/__init__.py +1 -0
  206. wbcore/contrib/currency/viewsets/menu/currency.py +8 -0
  207. wbcore/contrib/currency/viewsets/preview/__init__.py +1 -0
  208. wbcore/contrib/currency/viewsets/preview/currency.py +10 -0
  209. wbcore/contrib/currency/viewsets/titles/__init__.py +2 -0
  210. wbcore/contrib/currency/viewsets/titles/currency.py +6 -0
  211. wbcore/contrib/currency/viewsets/titles/currency_fx_rates.py +8 -0
  212. wbcore/contrib/dataloader/__init__.py +1 -0
  213. wbcore/contrib/dataloader/apps.py +5 -0
  214. wbcore/contrib/dataloader/dataloaders/__init__.py +2 -0
  215. wbcore/contrib/dataloader/dataloaders/dataloaders.py +24 -0
  216. wbcore/contrib/dataloader/dataloaders/proxies.py +81 -0
  217. wbcore/contrib/dataloader/models/__init__.py +1 -0
  218. wbcore/contrib/dataloader/models/entities.py +29 -0
  219. wbcore/contrib/dataloader/models/querysets/__init__.py +1 -0
  220. wbcore/contrib/dataloader/models/querysets/entities.py +27 -0
  221. wbcore/contrib/dataloader/tests/__init__.py +0 -0
  222. wbcore/contrib/dataloader/tests/conftest.py +26 -0
  223. wbcore/contrib/dataloader/tests/test/__init__.py +0 -0
  224. wbcore/contrib/dataloader/tests/test/dataloaders/__init__.py +0 -0
  225. wbcore/contrib/dataloader/tests/test/dataloaders/dataloaders.py +18 -0
  226. wbcore/contrib/dataloader/tests/test/dataloaders/protocols.py +6 -0
  227. wbcore/contrib/dataloader/tests/test/dataloaders/proxies.py +11 -0
  228. wbcore/contrib/dataloader/tests/test/factories.py +13 -0
  229. wbcore/contrib/dataloader/tests/test/models.py +10 -0
  230. wbcore/contrib/dataloader/tests/test_dataloaders.py +37 -0
  231. wbcore/contrib/dataloader/tests/test_entities.py +12 -0
  232. wbcore/contrib/dataloader/utils.py +17 -0
  233. wbcore/contrib/directory/__init__.py +0 -0
  234. wbcore/contrib/directory/admin/__init__.py +14 -0
  235. wbcore/contrib/directory/admin/contacts.py +54 -0
  236. wbcore/contrib/directory/admin/entries.py +230 -0
  237. wbcore/contrib/directory/admin/relationships.py +73 -0
  238. wbcore/contrib/directory/apps.py +6 -0
  239. wbcore/contrib/directory/configs.py +8 -0
  240. wbcore/contrib/directory/configurations.py +69 -0
  241. wbcore/contrib/directory/dynamic_preferences_registry.py +61 -0
  242. wbcore/contrib/directory/factories/__init__.py +34 -0
  243. wbcore/contrib/directory/factories/contacts.py +71 -0
  244. wbcore/contrib/directory/factories/entries.py +169 -0
  245. wbcore/contrib/directory/factories/relationships.py +77 -0
  246. wbcore/contrib/directory/filters/__init__.py +22 -0
  247. wbcore/contrib/directory/filters/contacts.py +233 -0
  248. wbcore/contrib/directory/filters/entries.py +247 -0
  249. wbcore/contrib/directory/filters/relationships.py +89 -0
  250. wbcore/contrib/directory/migrations/0001_initial.py +871 -0
  251. wbcore/contrib/directory/migrations/0002_auto_20230414_1553.py +42 -0
  252. wbcore/contrib/directory/migrations/0003_remove_entry_last_event.py +16 -0
  253. wbcore/contrib/directory/migrations/0004_entry_is_draft_entry.py +21 -0
  254. wbcore/contrib/directory/migrations/0005_entry_salutation.py +17 -0
  255. wbcore/contrib/directory/migrations/0006_employeremployeerelationship_position_name.py +23 -0
  256. wbcore/contrib/directory/migrations/0007_alter_bankingcontact_options.py +20 -0
  257. wbcore/contrib/directory/migrations/0008_bankingcontact_access.py +17 -0
  258. wbcore/contrib/directory/migrations/0009_remove_entry_external_identfier_and_more.py +22 -0
  259. wbcore/contrib/directory/migrations/__init__.py +0 -0
  260. wbcore/contrib/directory/models/__init__.py +27 -0
  261. wbcore/contrib/directory/models/contacts.py +554 -0
  262. wbcore/contrib/directory/models/entries.py +889 -0
  263. wbcore/contrib/directory/models/relationships.py +634 -0
  264. wbcore/contrib/directory/preferences.py +10 -0
  265. wbcore/contrib/directory/release_notes/1_0_0.md +13 -0
  266. wbcore/contrib/directory/release_notes/1_0_1.md +13 -0
  267. wbcore/contrib/directory/release_notes/__init__.py +0 -0
  268. wbcore/contrib/directory/serializers/__init__.py +63 -0
  269. wbcore/contrib/directory/serializers/companies.py +158 -0
  270. wbcore/contrib/directory/serializers/contacts.py +404 -0
  271. wbcore/contrib/directory/serializers/entries.py +344 -0
  272. wbcore/contrib/directory/serializers/entry_representations.py +35 -0
  273. wbcore/contrib/directory/serializers/persons.py +209 -0
  274. wbcore/contrib/directory/serializers/relationships.py +332 -0
  275. wbcore/contrib/directory/signals.py +3 -0
  276. wbcore/contrib/directory/tests/__init__.py +0 -0
  277. wbcore/contrib/directory/tests/conftest.py +60 -0
  278. wbcore/contrib/directory/tests/disable_signals.py +52 -0
  279. wbcore/contrib/directory/tests/e2e/__init__.py +7 -0
  280. wbcore/contrib/directory/tests/e2e/e2e_directory_utility.py +162 -0
  281. wbcore/contrib/directory/tests/signals.py +90 -0
  282. wbcore/contrib/directory/tests/test_configs.py +6 -0
  283. wbcore/contrib/directory/tests/test_filters.py +59 -0
  284. wbcore/contrib/directory/tests/test_models.py +285 -0
  285. wbcore/contrib/directory/tests/test_permissions.py +122 -0
  286. wbcore/contrib/directory/tests/test_serializers.py +201 -0
  287. wbcore/contrib/directory/tests/test_viewsets.py +677 -0
  288. wbcore/contrib/directory/tests/tests.py +130 -0
  289. wbcore/contrib/directory/typings.py +17 -0
  290. wbcore/contrib/directory/urls.py +135 -0
  291. wbcore/contrib/directory/viewsets/__init__.py +58 -0
  292. wbcore/contrib/directory/viewsets/buttons/__init__.py +7 -0
  293. wbcore/contrib/directory/viewsets/buttons/contacts.py +16 -0
  294. wbcore/contrib/directory/viewsets/buttons/entries.py +133 -0
  295. wbcore/contrib/directory/viewsets/buttons/relationships.py +31 -0
  296. wbcore/contrib/directory/viewsets/contacts.py +393 -0
  297. wbcore/contrib/directory/viewsets/display/__init__.py +36 -0
  298. wbcore/contrib/directory/viewsets/display/contacts.py +380 -0
  299. wbcore/contrib/directory/viewsets/display/entries.py +501 -0
  300. wbcore/contrib/directory/viewsets/display/relationships.py +315 -0
  301. wbcore/contrib/directory/viewsets/display/utils.py +33 -0
  302. wbcore/contrib/directory/viewsets/endpoints/__init__.py +22 -0
  303. wbcore/contrib/directory/viewsets/endpoints/contacts.py +54 -0
  304. wbcore/contrib/directory/viewsets/endpoints/entries.py +30 -0
  305. wbcore/contrib/directory/viewsets/endpoints/relationships.py +105 -0
  306. wbcore/contrib/directory/viewsets/entries.py +212 -0
  307. wbcore/contrib/directory/viewsets/menu/__init__.py +16 -0
  308. wbcore/contrib/directory/viewsets/menu/contacts.py +25 -0
  309. wbcore/contrib/directory/viewsets/menu/entries.py +61 -0
  310. wbcore/contrib/directory/viewsets/menu/relationships.py +31 -0
  311. wbcore/contrib/directory/viewsets/menu/utils.py +72 -0
  312. wbcore/contrib/directory/viewsets/mixins.py +11 -0
  313. wbcore/contrib/directory/viewsets/previews/__init__.py +2 -0
  314. wbcore/contrib/directory/viewsets/previews/contacts.py +18 -0
  315. wbcore/contrib/directory/viewsets/previews/entries.py +31 -0
  316. wbcore/contrib/directory/viewsets/relationships.py +241 -0
  317. wbcore/contrib/directory/viewsets/titles/__init__.py +30 -0
  318. wbcore/contrib/directory/viewsets/titles/contacts.py +122 -0
  319. wbcore/contrib/directory/viewsets/titles/entries.py +29 -0
  320. wbcore/contrib/directory/viewsets/titles/relationships.py +86 -0
  321. wbcore/contrib/directory/viewsets/titles/utils.py +46 -0
  322. wbcore/contrib/directory/viewsets/utils.py +101 -0
  323. wbcore/contrib/documents/__init__.py +0 -0
  324. wbcore/contrib/documents/admin.py +79 -0
  325. wbcore/contrib/documents/apps.py +6 -0
  326. wbcore/contrib/documents/factories.py +43 -0
  327. wbcore/contrib/documents/filters.py +82 -0
  328. wbcore/contrib/documents/migrations/0001_initial.py +189 -0
  329. wbcore/contrib/documents/migrations/0002_documentmodelrelationship_primary_and_more.py +22 -0
  330. wbcore/contrib/documents/migrations/0003_alter_documentmodelrelationship_unique_together_and_more.py +30 -0
  331. wbcore/contrib/documents/migrations/0004_auto_20240103_0958.py +44 -0
  332. wbcore/contrib/documents/migrations/0005_document_valid_from_document_valid_until_and_more.py +32 -0
  333. wbcore/contrib/documents/migrations/__init__.py +0 -0
  334. wbcore/contrib/documents/models/__init__.py +4 -0
  335. wbcore/contrib/documents/models/document_model_relationships.py +57 -0
  336. wbcore/contrib/documents/models/document_types.py +35 -0
  337. wbcore/contrib/documents/models/documents.py +309 -0
  338. wbcore/contrib/documents/models/mixins.py +10 -0
  339. wbcore/contrib/documents/models/shareable_links.py +90 -0
  340. wbcore/contrib/documents/release_notes/1_0_0.md +13 -0
  341. wbcore/contrib/documents/release_notes/__init__.py +0 -0
  342. wbcore/contrib/documents/serializers/__init__.py +12 -0
  343. wbcore/contrib/documents/serializers/document_model_relationships.py +31 -0
  344. wbcore/contrib/documents/serializers/document_types.py +40 -0
  345. wbcore/contrib/documents/serializers/documents.py +64 -0
  346. wbcore/contrib/documents/serializers/shareable_links.py +90 -0
  347. wbcore/contrib/documents/urls.py +42 -0
  348. wbcore/contrib/documents/viewsets/__init__.py +10 -0
  349. wbcore/contrib/documents/viewsets/buttons/__init__.py +3 -0
  350. wbcore/contrib/documents/viewsets/buttons/documents.py +53 -0
  351. wbcore/contrib/documents/viewsets/buttons/shareable_links.py +20 -0
  352. wbcore/contrib/documents/viewsets/buttons/signals.py +13 -0
  353. wbcore/contrib/documents/viewsets/display/__init__.py +4 -0
  354. wbcore/contrib/documents/viewsets/display/document_model_relationships.py +18 -0
  355. wbcore/contrib/documents/viewsets/display/document_types.py +23 -0
  356. wbcore/contrib/documents/viewsets/display/documents.py +101 -0
  357. wbcore/contrib/documents/viewsets/display/shareable_links.py +65 -0
  358. wbcore/contrib/documents/viewsets/document_model_relationships.py +25 -0
  359. wbcore/contrib/documents/viewsets/document_types.py +37 -0
  360. wbcore/contrib/documents/viewsets/documents.py +107 -0
  361. wbcore/contrib/documents/viewsets/endpoints/__init__.py +5 -0
  362. wbcore/contrib/documents/viewsets/endpoints/documents.py +19 -0
  363. wbcore/contrib/documents/viewsets/endpoints/documents_model_relationships.py +13 -0
  364. wbcore/contrib/documents/viewsets/endpoints/shareable_links.py +45 -0
  365. wbcore/contrib/documents/viewsets/menu/__init__.py +1 -0
  366. wbcore/contrib/documents/viewsets/menu/documents.py +24 -0
  367. wbcore/contrib/documents/viewsets/previews/__init__.py +1 -0
  368. wbcore/contrib/documents/viewsets/previews/documents.py +10 -0
  369. wbcore/contrib/documents/viewsets/shareable_links.py +109 -0
  370. wbcore/contrib/documents/viewsets/titles/__init__.py +3 -0
  371. wbcore/contrib/documents/viewsets/titles/document_types.py +13 -0
  372. wbcore/contrib/documents/viewsets/titles/documents.py +26 -0
  373. wbcore/contrib/documents/viewsets/titles/shareable_links.py +13 -0
  374. wbcore/contrib/dynamic_preferences/__init__.py +0 -0
  375. wbcore/contrib/dynamic_preferences/types.py +19 -0
  376. wbcore/contrib/example_app/__init__.py +0 -0
  377. wbcore/contrib/example_app/admin.py +117 -0
  378. wbcore/contrib/example_app/apps.py +6 -0
  379. wbcore/contrib/example_app/factories/__init__.py +8 -0
  380. wbcore/contrib/example_app/factories/event.py +26 -0
  381. wbcore/contrib/example_app/factories/league.py +15 -0
  382. wbcore/contrib/example_app/factories/match.py +25 -0
  383. wbcore/contrib/example_app/factories/person.py +32 -0
  384. wbcore/contrib/example_app/factories/role.py +9 -0
  385. wbcore/contrib/example_app/factories/sport.py +12 -0
  386. wbcore/contrib/example_app/factories/stadium.py +12 -0
  387. wbcore/contrib/example_app/factories/team.py +35 -0
  388. wbcore/contrib/example_app/filters/__init__.py +9 -0
  389. wbcore/contrib/example_app/filters/event.py +80 -0
  390. wbcore/contrib/example_app/filters/league.py +74 -0
  391. wbcore/contrib/example_app/filters/match.py +92 -0
  392. wbcore/contrib/example_app/filters/person.py +77 -0
  393. wbcore/contrib/example_app/filters/role.py +17 -0
  394. wbcore/contrib/example_app/filters/sport.py +26 -0
  395. wbcore/contrib/example_app/filters/stadium.py +30 -0
  396. wbcore/contrib/example_app/filters/team.py +77 -0
  397. wbcore/contrib/example_app/filters/teamresult.py +47 -0
  398. wbcore/contrib/example_app/migrations/0001_initial.py +498 -0
  399. wbcore/contrib/example_app/migrations/0002_sportperson_profile.py +49 -0
  400. wbcore/contrib/example_app/migrations/0003_change_stadium_capacity.py +31 -0
  401. wbcore/contrib/example_app/migrations/0004_alter_player_transfer_value.py +21 -0
  402. wbcore/contrib/example_app/migrations/0005_sportperson_profile_image.py +23 -0
  403. wbcore/contrib/example_app/migrations/0006_league_season_period_player_is_active_and_more.py +116 -0
  404. wbcore/contrib/example_app/migrations/0007_alter_player_options_alter_team_options_and_more.py +40 -0
  405. wbcore/contrib/example_app/migrations/__init__.py +0 -0
  406. wbcore/contrib/example_app/models.py +906 -0
  407. wbcore/contrib/example_app/serializers/__init__.py +31 -0
  408. wbcore/contrib/example_app/serializers/league.py +111 -0
  409. wbcore/contrib/example_app/serializers/match_event.py +304 -0
  410. wbcore/contrib/example_app/serializers/person_team.py +280 -0
  411. wbcore/contrib/example_app/serializers/role.py +16 -0
  412. wbcore/contrib/example_app/serializers/season.py +53 -0
  413. wbcore/contrib/example_app/serializers/sport.py +37 -0
  414. wbcore/contrib/example_app/serializers/stadium.py +58 -0
  415. wbcore/contrib/example_app/serializers/teamresult.py +43 -0
  416. wbcore/contrib/example_app/tests/__init__.py +0 -0
  417. wbcore/contrib/example_app/tests/conftest.py +13 -0
  418. wbcore/contrib/example_app/tests/e2e/__init__.py +1 -0
  419. wbcore/contrib/example_app/tests/e2e/e2e_example_app_utility.py +51 -0
  420. wbcore/contrib/example_app/tests/e2e/test_league.py +69 -0
  421. wbcore/contrib/example_app/tests/e2e/test_person.py +67 -0
  422. wbcore/contrib/example_app/tests/e2e/test_teams.py +56 -0
  423. wbcore/contrib/example_app/tests/signals.py +7 -0
  424. wbcore/contrib/example_app/tests/test_displays.py +40 -0
  425. wbcore/contrib/example_app/tests/test_filters.py +70 -0
  426. wbcore/contrib/example_app/tests/test_utils.py +25 -0
  427. wbcore/contrib/example_app/urls.py +79 -0
  428. wbcore/contrib/example_app/utils.py +20 -0
  429. wbcore/contrib/example_app/viewsets/__init__.py +29 -0
  430. wbcore/contrib/example_app/viewsets/buttons/__init__.py +2 -0
  431. wbcore/contrib/example_app/viewsets/buttons/person.py +18 -0
  432. wbcore/contrib/example_app/viewsets/buttons/team.py +23 -0
  433. wbcore/contrib/example_app/viewsets/displays/__init__.py +23 -0
  434. wbcore/contrib/example_app/viewsets/displays/event.py +186 -0
  435. wbcore/contrib/example_app/viewsets/displays/league.py +390 -0
  436. wbcore/contrib/example_app/viewsets/displays/match.py +264 -0
  437. wbcore/contrib/example_app/viewsets/displays/person.py +251 -0
  438. wbcore/contrib/example_app/viewsets/displays/role.py +16 -0
  439. wbcore/contrib/example_app/viewsets/displays/season.py +65 -0
  440. wbcore/contrib/example_app/viewsets/displays/sport.py +115 -0
  441. wbcore/contrib/example_app/viewsets/displays/stadium.py +149 -0
  442. wbcore/contrib/example_app/viewsets/displays/team.py +230 -0
  443. wbcore/contrib/example_app/viewsets/displays/teamresult.py +30 -0
  444. wbcore/contrib/example_app/viewsets/endpoints/__init__.py +12 -0
  445. wbcore/contrib/example_app/viewsets/endpoints/endpoints.py +79 -0
  446. wbcore/contrib/example_app/viewsets/event.py +287 -0
  447. wbcore/contrib/example_app/viewsets/league.py +72 -0
  448. wbcore/contrib/example_app/viewsets/match.py +119 -0
  449. wbcore/contrib/example_app/viewsets/menu/__init__.py +1 -0
  450. wbcore/contrib/example_app/viewsets/menu/menus.py +102 -0
  451. wbcore/contrib/example_app/viewsets/menus.py +63 -0
  452. wbcore/contrib/example_app/viewsets/person.py +133 -0
  453. wbcore/contrib/example_app/viewsets/role.py +25 -0
  454. wbcore/contrib/example_app/viewsets/season.py +23 -0
  455. wbcore/contrib/example_app/viewsets/sport.py +26 -0
  456. wbcore/contrib/example_app/viewsets/stadium.py +30 -0
  457. wbcore/contrib/example_app/viewsets/team.py +68 -0
  458. wbcore/contrib/example_app/viewsets/teamresult.py +106 -0
  459. wbcore/contrib/example_app/viewsets/titles/__init__.py +9 -0
  460. wbcore/contrib/example_app/viewsets/titles/event.py +41 -0
  461. wbcore/contrib/example_app/viewsets/titles/league.py +24 -0
  462. wbcore/contrib/example_app/viewsets/titles/match.py +34 -0
  463. wbcore/contrib/example_app/viewsets/titles/person.py +35 -0
  464. wbcore/contrib/example_app/viewsets/titles/role.py +13 -0
  465. wbcore/contrib/example_app/viewsets/titles/sport.py +13 -0
  466. wbcore/contrib/example_app/viewsets/titles/stadium.py +13 -0
  467. wbcore/contrib/example_app/viewsets/titles/team.py +24 -0
  468. wbcore/contrib/example_app/viewsets/titles/teamresult.py +12 -0
  469. wbcore/contrib/geography/__init__.py +0 -0
  470. wbcore/contrib/geography/admin.py +30 -0
  471. wbcore/contrib/geography/apps.py +5 -0
  472. wbcore/contrib/geography/factories.py +46 -0
  473. wbcore/contrib/geography/import_export/__init__.py +0 -0
  474. wbcore/contrib/geography/import_export/resources/__init__.py +0 -0
  475. wbcore/contrib/geography/import_export/resources/geography.py +10 -0
  476. wbcore/contrib/geography/migrations/0001_initial.py +110 -0
  477. wbcore/contrib/geography/migrations/__init__.py +0 -0
  478. wbcore/contrib/geography/models.py +147 -0
  479. wbcore/contrib/geography/release_notes/1_0_0.md +13 -0
  480. wbcore/contrib/geography/release_notes/__init__.py +0 -0
  481. wbcore/contrib/geography/serializers.py +32 -0
  482. wbcore/contrib/geography/tests/__init__.py +0 -0
  483. wbcore/contrib/geography/tests/conftest.py +12 -0
  484. wbcore/contrib/geography/tests/signals.py +7 -0
  485. wbcore/contrib/geography/tests/test_models.py +12 -0
  486. wbcore/contrib/geography/tests/test_serializers.py +7 -0
  487. wbcore/contrib/geography/tests/test_viewsets.py +6 -0
  488. wbcore/contrib/geography/tests/tests.py +12 -0
  489. wbcore/contrib/geography/urls.py +12 -0
  490. wbcore/contrib/geography/viewsets/__init__.py +1 -0
  491. wbcore/contrib/geography/viewsets/buttons/__init__.py +0 -0
  492. wbcore/contrib/geography/viewsets/display/__init__.py +1 -0
  493. wbcore/contrib/geography/viewsets/display/geography.py +32 -0
  494. wbcore/contrib/geography/viewsets/endpoints/__init__.py +0 -0
  495. wbcore/contrib/geography/viewsets/geography.py +57 -0
  496. wbcore/contrib/geography/viewsets/menu/__init__.py +1 -0
  497. wbcore/contrib/geography/viewsets/menu/geography.py +8 -0
  498. wbcore/contrib/geography/viewsets/preview/__init__.py +1 -0
  499. wbcore/contrib/geography/viewsets/preview/geography.py +19 -0
  500. wbcore/contrib/geography/viewsets/titles/__init__.py +0 -0
  501. wbcore/contrib/geography/viewsets/titles/geography.py +14 -0
  502. wbcore/contrib/gleap/__init__.py +0 -0
  503. wbcore/contrib/gleap/apps.py +5 -0
  504. wbcore/contrib/gleap/configs.py +11 -0
  505. wbcore/contrib/gleap/configurations.py +6 -0
  506. wbcore/contrib/gleap/hashes.py +13 -0
  507. wbcore/contrib/gleap/tests/__init__.py +0 -0
  508. wbcore/contrib/gleap/tests/conftest.py +1 -0
  509. wbcore/contrib/gleap/tests/tests.py +29 -0
  510. wbcore/contrib/gleap/urls.py +7 -0
  511. wbcore/contrib/gleap/views.py +31 -0
  512. wbcore/contrib/guardian/migrations/0001_initial.py +103 -0
  513. wbcore/contrib/guardian/migrations/__init__.py +0 -0
  514. wbcore/contrib/guardian/models/__init__.py +1 -0
  515. wbcore/contrib/guardian/models/mixins.py +138 -0
  516. wbcore/contrib/guardian/models/models.py +29 -0
  517. wbcore/contrib/guardian/tests/__init__.py +0 -0
  518. wbcore/contrib/guardian/tests/conftest.py +1 -0
  519. wbcore/contrib/guardian/tests/test_model_mixins.py +93 -0
  520. wbcore/contrib/guardian/tests/test_tasks.py +77 -0
  521. wbcore/contrib/guardian/tests/test_utils.py +196 -0
  522. wbcore/contrib/guardian/tests/test_viewsets.py +48 -0
  523. wbcore/contrib/guardian/viewsets/__init__.py +1 -0
  524. wbcore/contrib/guardian/viewsets/configs/__init__.py +4 -0
  525. wbcore/contrib/guardian/viewsets/configs/buttons.py +27 -0
  526. wbcore/contrib/guardian/viewsets/configs/displays.py +43 -0
  527. wbcore/contrib/guardian/viewsets/configs/endpoints.py +18 -0
  528. wbcore/contrib/guardian/viewsets/configs/titles.py +16 -0
  529. wbcore/contrib/guardian/viewsets/mixins.py +6 -0
  530. wbcore/contrib/guardian/viewsets/viewsets.py +139 -0
  531. wbcore/contrib/icons/__init__.py +1 -0
  532. wbcore/contrib/icons/apps.py +5 -0
  533. wbcore/contrib/icons/backends/__init__.py +5 -0
  534. wbcore/contrib/icons/backends/default.py +374 -0
  535. wbcore/contrib/icons/backends/material.py +135 -0
  536. wbcore/contrib/icons/icons.py +169 -0
  537. wbcore/contrib/icons/models.py +11 -0
  538. wbcore/contrib/icons/serializers.py +18 -0
  539. wbcore/contrib/io/__init__.py +0 -0
  540. wbcore/contrib/io/admin.py +150 -0
  541. wbcore/contrib/io/apps.py +27 -0
  542. wbcore/contrib/io/backends/__init__.py +5 -0
  543. wbcore/contrib/io/backends/abstract.py +60 -0
  544. wbcore/contrib/io/backends/mail.py +85 -0
  545. wbcore/contrib/io/backends/sftp.py +63 -0
  546. wbcore/contrib/io/backends/stream.py +92 -0
  547. wbcore/contrib/io/backends/utils.py +42 -0
  548. wbcore/contrib/io/configs/__init__.py +0 -0
  549. wbcore/contrib/io/configs/endpoints.py +12 -0
  550. wbcore/contrib/io/configurations/__init__.py +1 -0
  551. wbcore/contrib/io/configurations/base.py +7 -0
  552. wbcore/contrib/io/dynamic_preferences_registry.py +15 -0
  553. wbcore/contrib/io/enums.py +15 -0
  554. wbcore/contrib/io/exceptions.py +16 -0
  555. wbcore/contrib/io/factories.py +202 -0
  556. wbcore/contrib/io/imports.py +305 -0
  557. wbcore/contrib/io/management/__init__.py +13 -0
  558. wbcore/contrib/io/migrations/0001_initial_squashed.py +319 -0
  559. wbcore/contrib/io/migrations/0002_importsource_creator.py +26 -0
  560. wbcore/contrib/io/migrations/0003_auto_20240103_1000.py +46 -0
  561. wbcore/contrib/io/migrations/0004_alter_importsource_status_exportsource.py +134 -0
  562. wbcore/contrib/io/migrations/0005_exportsource_data_alter_exportsource_query_str_and_more.py +67 -0
  563. wbcore/contrib/io/migrations/0006_alter_exportsource_query_params.py +20 -0
  564. wbcore/contrib/io/migrations/0007_alter_exportsource_query_params.py +23 -0
  565. wbcore/contrib/io/migrations/__init__.py +0 -0
  566. wbcore/contrib/io/mixins.py +37 -0
  567. wbcore/contrib/io/models.py +1005 -0
  568. wbcore/contrib/io/release_notes/1_0_0.md +13 -0
  569. wbcore/contrib/io/release_notes/__init__.py +0 -0
  570. wbcore/contrib/io/resources.py +177 -0
  571. wbcore/contrib/io/serializers.py +136 -0
  572. wbcore/contrib/io/tests/__init__.py +0 -0
  573. wbcore/contrib/io/tests/conftest.py +42 -0
  574. wbcore/contrib/io/tests/test_backends.py +128 -0
  575. wbcore/contrib/io/tests/test_exports.py +130 -0
  576. wbcore/contrib/io/tests/test_imports.py +167 -0
  577. wbcore/contrib/io/tests/test_models.py +363 -0
  578. wbcore/contrib/io/tests/tests.py +18 -0
  579. wbcore/contrib/io/urls.py +28 -0
  580. wbcore/contrib/io/utils.py +44 -0
  581. wbcore/contrib/io/viewset_mixins.py +301 -0
  582. wbcore/contrib/io/viewsets.py +138 -0
  583. wbcore/contrib/notifications/__init__.py +0 -0
  584. wbcore/contrib/notifications/admin.py +61 -0
  585. wbcore/contrib/notifications/apps.py +38 -0
  586. wbcore/contrib/notifications/backends/__init__.py +0 -0
  587. wbcore/contrib/notifications/backends/abstract_backend.py +13 -0
  588. wbcore/contrib/notifications/backends/console/__init__.py +1 -0
  589. wbcore/contrib/notifications/backends/console/backends.py +25 -0
  590. wbcore/contrib/notifications/backends/firebase/__init__.py +1 -0
  591. wbcore/contrib/notifications/backends/firebase/backends.py +102 -0
  592. wbcore/contrib/notifications/configs.py +12 -0
  593. wbcore/contrib/notifications/configurations.py +9 -0
  594. wbcore/contrib/notifications/dispatch.py +71 -0
  595. wbcore/contrib/notifications/factories/__init__.py +0 -0
  596. wbcore/contrib/notifications/factories/notification_types.py +22 -0
  597. wbcore/contrib/notifications/factories/notifications.py +16 -0
  598. wbcore/contrib/notifications/factories/tokens.py +11 -0
  599. wbcore/contrib/notifications/migrations/0001_initial.py +116 -0
  600. wbcore/contrib/notifications/migrations/0002_notificationusertoken_unique_user_token_device.py +18 -0
  601. wbcore/contrib/notifications/migrations/0003_notificationusertoken_updated.py +17 -0
  602. wbcore/contrib/notifications/migrations/0004_alter_notification_body.py +17 -0
  603. wbcore/contrib/notifications/migrations/0005_alter_notification_endpoint.py +17 -0
  604. wbcore/contrib/notifications/migrations/0006_notification_created.py +25 -0
  605. wbcore/contrib/notifications/migrations/__init__.py +0 -0
  606. wbcore/contrib/notifications/models/__init__.py +3 -0
  607. wbcore/contrib/notifications/models/notification_types.py +112 -0
  608. wbcore/contrib/notifications/models/notifications.py +81 -0
  609. wbcore/contrib/notifications/models/tokens.py +45 -0
  610. wbcore/contrib/notifications/release_notes/1_0_0.md +13 -0
  611. wbcore/contrib/notifications/release_notes/__init__.py +0 -0
  612. wbcore/contrib/notifications/serializers/__init__.py +5 -0
  613. wbcore/contrib/notifications/serializers/notification_types.py +36 -0
  614. wbcore/contrib/notifications/serializers/notifications.py +33 -0
  615. wbcore/contrib/notifications/tasks.py +55 -0
  616. wbcore/contrib/notifications/tests/__init__.py +0 -0
  617. wbcore/contrib/notifications/tests/conftest.py +47 -0
  618. wbcore/contrib/notifications/tests/test_backends/__init__.py +0 -0
  619. wbcore/contrib/notifications/tests/test_backends/test_firebase.py +78 -0
  620. wbcore/contrib/notifications/tests/test_configs.py +7 -0
  621. wbcore/contrib/notifications/tests/test_models/__init__.py +0 -0
  622. wbcore/contrib/notifications/tests/test_models/test_notification_types.py +84 -0
  623. wbcore/contrib/notifications/tests/test_models/test_notifications.py +45 -0
  624. wbcore/contrib/notifications/tests/test_models/test_tokens.py +21 -0
  625. wbcore/contrib/notifications/tests/test_serializers/__init__.py +0 -0
  626. wbcore/contrib/notifications/tests/test_serializers/test_notification_types.py +53 -0
  627. wbcore/contrib/notifications/tests/test_serializers/test_notifications.py +23 -0
  628. wbcore/contrib/notifications/tests/test_tasks.py +71 -0
  629. wbcore/contrib/notifications/tests/test_utils.py +0 -0
  630. wbcore/contrib/notifications/tests/test_viewsets/__init__.py +0 -0
  631. wbcore/contrib/notifications/tests/test_viewsets/test_notification_types.py +121 -0
  632. wbcore/contrib/notifications/tests/test_viewsets/test_notifications.py +120 -0
  633. wbcore/contrib/notifications/urls.py +26 -0
  634. wbcore/contrib/notifications/utils.py +20 -0
  635. wbcore/contrib/notifications/views.py +62 -0
  636. wbcore/contrib/notifications/viewsets/__init__.py +5 -0
  637. wbcore/contrib/notifications/viewsets/menus.py +14 -0
  638. wbcore/contrib/notifications/viewsets/notification_types.py +40 -0
  639. wbcore/contrib/notifications/viewsets/notifications.py +57 -0
  640. wbcore/contrib/tags/__init__.py +0 -0
  641. wbcore/contrib/tags/admin.py +17 -0
  642. wbcore/contrib/tags/apps.py +9 -0
  643. wbcore/contrib/tags/factories.py +27 -0
  644. wbcore/contrib/tags/filters.py +42 -0
  645. wbcore/contrib/tags/migrations/0001_initial.py +62 -0
  646. wbcore/contrib/tags/migrations/__init__.py +0 -0
  647. wbcore/contrib/tags/models/__init__.py +2 -0
  648. wbcore/contrib/tags/models/mixins.py +27 -0
  649. wbcore/contrib/tags/models/tags.py +103 -0
  650. wbcore/contrib/tags/release_notes/1_0_0.md +13 -0
  651. wbcore/contrib/tags/release_notes/__init__.py +0 -0
  652. wbcore/contrib/tags/serializers.py +87 -0
  653. wbcore/contrib/tags/signals.py +17 -0
  654. wbcore/contrib/tags/tests/__init__.py +0 -0
  655. wbcore/contrib/tags/tests/conftest.py +7 -0
  656. wbcore/contrib/tags/tests/tests.py +17 -0
  657. wbcore/contrib/tags/urls.py +14 -0
  658. wbcore/contrib/tags/viewsets/__init__.py +8 -0
  659. wbcore/contrib/tags/viewsets/display.py +50 -0
  660. wbcore/contrib/tags/viewsets/menu.py +15 -0
  661. wbcore/contrib/tags/viewsets/viewsets.py +58 -0
  662. wbcore/contrib/workflow/__init__.py +1 -0
  663. wbcore/contrib/workflow/admin/__init__.py +15 -0
  664. wbcore/contrib/workflow/admin/condition.py +8 -0
  665. wbcore/contrib/workflow/admin/data.py +14 -0
  666. wbcore/contrib/workflow/admin/display.py +8 -0
  667. wbcore/contrib/workflow/admin/process.py +36 -0
  668. wbcore/contrib/workflow/admin/step.py +91 -0
  669. wbcore/contrib/workflow/admin/transition.py +8 -0
  670. wbcore/contrib/workflow/admin/workflow.py +8 -0
  671. wbcore/contrib/workflow/apps.py +26 -0
  672. wbcore/contrib/workflow/configs.py +10 -0
  673. wbcore/contrib/workflow/decorators.py +25 -0
  674. wbcore/contrib/workflow/dispatch.py +23 -0
  675. wbcore/contrib/workflow/factories/__init__.py +18 -0
  676. wbcore/contrib/workflow/factories/condition.py +15 -0
  677. wbcore/contrib/workflow/factories/data.py +22 -0
  678. wbcore/contrib/workflow/factories/display.py +27 -0
  679. wbcore/contrib/workflow/factories/process.py +69 -0
  680. wbcore/contrib/workflow/factories/step.py +155 -0
  681. wbcore/contrib/workflow/factories/transition.py +25 -0
  682. wbcore/contrib/workflow/factories/workflow.py +19 -0
  683. wbcore/contrib/workflow/filters/__init__.py +24 -0
  684. wbcore/contrib/workflow/filters/condition.py +23 -0
  685. wbcore/contrib/workflow/filters/data.py +24 -0
  686. wbcore/contrib/workflow/filters/process.py +164 -0
  687. wbcore/contrib/workflow/filters/step.py +226 -0
  688. wbcore/contrib/workflow/filters/transition.py +41 -0
  689. wbcore/contrib/workflow/filters/workflow.py +43 -0
  690. wbcore/contrib/workflow/migrations/0001_initial.py +788 -0
  691. wbcore/contrib/workflow/migrations/0002_alter_step_step_type.py +31 -0
  692. wbcore/contrib/workflow/migrations/0003_alter_condition_attribute_name_and_more.py +75 -0
  693. wbcore/contrib/workflow/migrations/0004_alter_userstep_assignee_method.py +27 -0
  694. wbcore/contrib/workflow/migrations/0005_alter_userstep_assignee_method.py +17 -0
  695. wbcore/contrib/workflow/migrations/__init__.py +0 -0
  696. wbcore/contrib/workflow/models/__init__.py +19 -0
  697. wbcore/contrib/workflow/models/condition.py +123 -0
  698. wbcore/contrib/workflow/models/data.py +238 -0
  699. wbcore/contrib/workflow/models/display.py +33 -0
  700. wbcore/contrib/workflow/models/process.py +243 -0
  701. wbcore/contrib/workflow/models/step.py +735 -0
  702. wbcore/contrib/workflow/models/transition.py +70 -0
  703. wbcore/contrib/workflow/models/workflow.py +307 -0
  704. wbcore/contrib/workflow/serializers/__init__.py +37 -0
  705. wbcore/contrib/workflow/serializers/condition.py +64 -0
  706. wbcore/contrib/workflow/serializers/data.py +135 -0
  707. wbcore/contrib/workflow/serializers/display.py +25 -0
  708. wbcore/contrib/workflow/serializers/process.py +182 -0
  709. wbcore/contrib/workflow/serializers/signals.py +25 -0
  710. wbcore/contrib/workflow/serializers/step.py +364 -0
  711. wbcore/contrib/workflow/serializers/transition.py +80 -0
  712. wbcore/contrib/workflow/serializers/workflow.py +124 -0
  713. wbcore/contrib/workflow/sites.py +43 -0
  714. wbcore/contrib/workflow/tests/__init__.py +0 -0
  715. wbcore/contrib/workflow/tests/conftest.py +57 -0
  716. wbcore/contrib/workflow/tests/test_configs.py +6 -0
  717. wbcore/contrib/workflow/tests/test_dispatch.py +88 -0
  718. wbcore/contrib/workflow/tests/test_displays.py +119 -0
  719. wbcore/contrib/workflow/tests/test_filters.py +86 -0
  720. wbcore/contrib/workflow/tests/test_serializers.py +177 -0
  721. wbcore/contrib/workflow/tests/test_viewsets.py +358 -0
  722. wbcore/contrib/workflow/tests/test_workflow_assignees.py +225 -0
  723. wbcore/contrib/workflow/tests/tests.py +24 -0
  724. wbcore/contrib/workflow/urls.py +66 -0
  725. wbcore/contrib/workflow/utils.py +13 -0
  726. wbcore/contrib/workflow/viewsets/__init__.py +17 -0
  727. wbcore/contrib/workflow/viewsets/buttons/__init__.py +1 -0
  728. wbcore/contrib/workflow/viewsets/buttons/step.py +67 -0
  729. wbcore/contrib/workflow/viewsets/condition.py +32 -0
  730. wbcore/contrib/workflow/viewsets/data.py +33 -0
  731. wbcore/contrib/workflow/viewsets/display/__init__.py +15 -0
  732. wbcore/contrib/workflow/viewsets/display/condition.py +50 -0
  733. wbcore/contrib/workflow/viewsets/display/data.py +50 -0
  734. wbcore/contrib/workflow/viewsets/display/process.py +185 -0
  735. wbcore/contrib/workflow/viewsets/display/step.py +453 -0
  736. wbcore/contrib/workflow/viewsets/display/transition.py +75 -0
  737. wbcore/contrib/workflow/viewsets/display/workflow.py +167 -0
  738. wbcore/contrib/workflow/viewsets/endpoints/__init__.py +5 -0
  739. wbcore/contrib/workflow/viewsets/endpoints/condition.py +9 -0
  740. wbcore/contrib/workflow/viewsets/endpoints/data.py +9 -0
  741. wbcore/contrib/workflow/viewsets/endpoints/process.py +11 -0
  742. wbcore/contrib/workflow/viewsets/endpoints/step.py +24 -0
  743. wbcore/contrib/workflow/viewsets/endpoints/transition.py +18 -0
  744. wbcore/contrib/workflow/viewsets/menu/__init__.py +15 -0
  745. wbcore/contrib/workflow/viewsets/menu/condition.py +18 -0
  746. wbcore/contrib/workflow/viewsets/menu/data.py +18 -0
  747. wbcore/contrib/workflow/viewsets/menu/process.py +18 -0
  748. wbcore/contrib/workflow/viewsets/menu/step.py +123 -0
  749. wbcore/contrib/workflow/viewsets/menu/transition.py +18 -0
  750. wbcore/contrib/workflow/viewsets/menu/workflow.py +18 -0
  751. wbcore/contrib/workflow/viewsets/process.py +171 -0
  752. wbcore/contrib/workflow/viewsets/step.py +229 -0
  753. wbcore/contrib/workflow/viewsets/titles/__init__.py +16 -0
  754. wbcore/contrib/workflow/viewsets/titles/condition.py +13 -0
  755. wbcore/contrib/workflow/viewsets/titles/data.py +13 -0
  756. wbcore/contrib/workflow/viewsets/titles/process.py +26 -0
  757. wbcore/contrib/workflow/viewsets/titles/step.py +101 -0
  758. wbcore/contrib/workflow/viewsets/titles/transition.py +13 -0
  759. wbcore/contrib/workflow/viewsets/titles/workflow.py +13 -0
  760. wbcore/contrib/workflow/viewsets/transition.py +57 -0
  761. wbcore/contrib/workflow/viewsets/workflow.py +36 -0
  762. wbcore/contrib/workflow/workflows/__init__.py +1 -0
  763. wbcore/contrib/workflow/workflows/assignees.py +82 -0
  764. wbcore/crontab/__init__.py +0 -0
  765. wbcore/crontab/serializers.py +23 -0
  766. wbcore/crontab/viewsets.py +9 -0
  767. wbcore/dispatch.py +55 -0
  768. wbcore/docs/__init__.py +25 -0
  769. wbcore/docs/orderable.md +29 -0
  770. wbcore/docs/reparent.md +13 -0
  771. wbcore/dynamic_preferences_registry.py +29 -0
  772. wbcore/enums.py +98 -0
  773. wbcore/filters/__init__.py +21 -0
  774. wbcore/filters/backends.py +19 -0
  775. wbcore/filters/defaults.py +69 -0
  776. wbcore/filters/fields/__init__.py +15 -0
  777. wbcore/filters/fields/booleans.py +6 -0
  778. wbcore/filters/fields/choices.py +61 -0
  779. wbcore/filters/fields/content_type.py +33 -0
  780. wbcore/filters/fields/datetime.py +91 -0
  781. wbcore/filters/fields/models.py +113 -0
  782. wbcore/filters/fields/multiple_lookups.py +19 -0
  783. wbcore/filters/fields/numbers.py +46 -0
  784. wbcore/filters/fields/text.py +6 -0
  785. wbcore/filters/filterset.py +217 -0
  786. wbcore/filters/lookups.py +41 -0
  787. wbcore/filters/mixins.py +112 -0
  788. wbcore/filters/utils.py +21 -0
  789. wbcore/forms.py +125 -0
  790. wbcore/frontend.py +23 -0
  791. wbcore/frontend_user_configuration.py +96 -0
  792. wbcore/fsm/__init__.py +0 -0
  793. wbcore/fsm/markdown_extensions.py +31 -0
  794. wbcore/fsm/mixins.py +114 -0
  795. wbcore/management/__init__.py +88 -0
  796. wbcore/management/commands/__init__.py +0 -0
  797. wbcore/management/commands/bootstrap.py +18 -0
  798. wbcore/management/commands/clean_obsolete_object.py +21 -0
  799. wbcore/management/commands/handle_release_notes.py +53 -0
  800. wbcore/markdown/__init__.py +1 -0
  801. wbcore/markdown/admin.py +9 -0
  802. wbcore/markdown/dynamic_preferences_registry.py +16 -0
  803. wbcore/markdown/models.py +38 -0
  804. wbcore/markdown/template.py +38 -0
  805. wbcore/markdown/utils.py +36 -0
  806. wbcore/markdown/views.py +67 -0
  807. wbcore/menus/__init__.py +2 -0
  808. wbcore/menus/menus.py +96 -0
  809. wbcore/menus/registry.py +28 -0
  810. wbcore/menus/views.py +41 -0
  811. wbcore/messages.py +51 -0
  812. wbcore/metadata/__init__.py +0 -0
  813. wbcore/metadata/configs/__init__.py +0 -0
  814. wbcore/metadata/configs/base.py +86 -0
  815. wbcore/metadata/configs/buttons/__init__.py +4 -0
  816. wbcore/metadata/configs/buttons/bases.py +91 -0
  817. wbcore/metadata/configs/buttons/buttons.py +142 -0
  818. wbcore/metadata/configs/buttons/enums.py +25 -0
  819. wbcore/metadata/configs/buttons/metadata.py +8 -0
  820. wbcore/metadata/configs/buttons/view_config.py +176 -0
  821. wbcore/metadata/configs/display/__init__.py +4 -0
  822. wbcore/metadata/configs/display/configs.py +15 -0
  823. wbcore/metadata/configs/display/display.py +234 -0
  824. wbcore/metadata/configs/display/formatting.py +50 -0
  825. wbcore/metadata/configs/display/instance_display/__init__.py +15 -0
  826. wbcore/metadata/configs/display/instance_display/display.py +37 -0
  827. wbcore/metadata/configs/display/instance_display/enums.py +7 -0
  828. wbcore/metadata/configs/display/instance_display/layouts/__init__.py +3 -0
  829. wbcore/metadata/configs/display/instance_display/layouts/inlines.py +58 -0
  830. wbcore/metadata/configs/display/instance_display/layouts/layouts.py +77 -0
  831. wbcore/metadata/configs/display/instance_display/layouts/sections.py +43 -0
  832. wbcore/metadata/configs/display/instance_display/operators.py +22 -0
  833. wbcore/metadata/configs/display/instance_display/pages.py +40 -0
  834. wbcore/metadata/configs/display/instance_display/shortcuts.py +70 -0
  835. wbcore/metadata/configs/display/instance_display/signals.py +3 -0
  836. wbcore/metadata/configs/display/instance_display/styles.py +42 -0
  837. wbcore/metadata/configs/display/instance_display/utils.py +74 -0
  838. wbcore/metadata/configs/display/list_display.py +230 -0
  839. wbcore/metadata/configs/display/models.py +20 -0
  840. wbcore/metadata/configs/display/view_config.py +82 -0
  841. wbcore/metadata/configs/display/views.py +48 -0
  842. wbcore/metadata/configs/display/windows.py +28 -0
  843. wbcore/metadata/configs/documentations.py +11 -0
  844. wbcore/metadata/configs/endpoints.py +176 -0
  845. wbcore/metadata/configs/fields.py +17 -0
  846. wbcore/metadata/configs/filter_fields.py +42 -0
  847. wbcore/metadata/configs/identifiers.py +25 -0
  848. wbcore/metadata/configs/ordering_fields.py +24 -0
  849. wbcore/metadata/configs/paginations.py +11 -0
  850. wbcore/metadata/configs/preview.py +41 -0
  851. wbcore/metadata/configs/primary_keys.py +9 -0
  852. wbcore/metadata/configs/search_fields.py +9 -0
  853. wbcore/metadata/configs/titles.py +48 -0
  854. wbcore/metadata/configs/window_types.py +13 -0
  855. wbcore/metadata/exceptions.py +0 -0
  856. wbcore/metadata/metadata.py +33 -0
  857. wbcore/metadata/mixins.py +105 -0
  858. wbcore/metadata/tests/__init__.py +0 -0
  859. wbcore/metadata/tests/test_buttons.py +179 -0
  860. wbcore/metadata/utils.py +4 -0
  861. wbcore/migrations/0001_initial_squashed_squashed_0010_preset_appliedpreset.py +398 -0
  862. wbcore/migrations/0011_genericmodel.py +22 -0
  863. wbcore/migrations/0012_delete_notification.py +15 -0
  864. wbcore/migrations/0013_delete_colorgradient.py +14 -0
  865. wbcore/migrations/0014_biguserobjectpermission_system.py +44 -0
  866. wbcore/migrations/__init__.py +0 -0
  867. wbcore/models/__init__.py +6 -0
  868. wbcore/models/base.py +182 -0
  869. wbcore/models/fields.py +29 -0
  870. wbcore/models/orderable.py +6 -0
  871. wbcore/pagination.py +65 -0
  872. wbcore/pandas/__init__.py +0 -0
  873. wbcore/pandas/fields.py +136 -0
  874. wbcore/pandas/filters.py +113 -0
  875. wbcore/pandas/filterset.py +26 -0
  876. wbcore/pandas/metadata.py +14 -0
  877. wbcore/pandas/utils.py +129 -0
  878. wbcore/pandas/views.py +157 -0
  879. wbcore/permissions/__init__.py +0 -0
  880. wbcore/permissions/backend.py +35 -0
  881. wbcore/permissions/mixins.py +72 -0
  882. wbcore/permissions/permissions.py +50 -0
  883. wbcore/permissions/registry.py +32 -0
  884. wbcore/permissions/shortcuts.py +37 -0
  885. wbcore/permissions/utils.py +26 -0
  886. wbcore/release_notes/__init__.py +0 -0
  887. wbcore/release_notes/admin.py +30 -0
  888. wbcore/release_notes/buttons.py +25 -0
  889. wbcore/release_notes/display.py +57 -0
  890. wbcore/release_notes/filters.py +35 -0
  891. wbcore/release_notes/models.py +65 -0
  892. wbcore/release_notes/serializers.py +19 -0
  893. wbcore/release_notes/utils.py +14 -0
  894. wbcore/release_notes/viewsets.py +45 -0
  895. wbcore/reversion/__init__.py +0 -0
  896. wbcore/reversion/filters.py +38 -0
  897. wbcore/reversion/serializers.py +80 -0
  898. wbcore/reversion/viewsets/__init__.py +6 -0
  899. wbcore/reversion/viewsets/buttons.py +94 -0
  900. wbcore/reversion/viewsets/displays.py +44 -0
  901. wbcore/reversion/viewsets/endpoints.py +29 -0
  902. wbcore/reversion/viewsets/titles.py +17 -0
  903. wbcore/reversion/viewsets/viewsets.py +111 -0
  904. wbcore/routers.py +63 -0
  905. wbcore/search/__init__.py +62 -0
  906. wbcore/serializers/__init__.py +66 -0
  907. wbcore/serializers/fields/__init__.py +58 -0
  908. wbcore/serializers/fields/boolean.py +51 -0
  909. wbcore/serializers/fields/choice.py +36 -0
  910. wbcore/serializers/fields/datetime.py +143 -0
  911. wbcore/serializers/fields/fields.py +190 -0
  912. wbcore/serializers/fields/file.py +20 -0
  913. wbcore/serializers/fields/fsm.py +20 -0
  914. wbcore/serializers/fields/json.py +56 -0
  915. wbcore/serializers/fields/list.py +105 -0
  916. wbcore/serializers/fields/mixins.py +175 -0
  917. wbcore/serializers/fields/number.py +103 -0
  918. wbcore/serializers/fields/other.py +42 -0
  919. wbcore/serializers/fields/primary_key.py +22 -0
  920. wbcore/serializers/fields/related.py +152 -0
  921. wbcore/serializers/fields/text.py +136 -0
  922. wbcore/serializers/fields/types.py +41 -0
  923. wbcore/serializers/mixins.py +24 -0
  924. wbcore/serializers/serializers.py +415 -0
  925. wbcore/serializers/utils.py +142 -0
  926. wbcore/shares/__init__.py +1 -0
  927. wbcore/shares/config.py +63 -0
  928. wbcore/shares/decorator.py +13 -0
  929. wbcore/shares/signals.py +3 -0
  930. wbcore/shares/sites.py +28 -0
  931. wbcore/shares/views.py +24 -0
  932. wbcore/signals/__init__.py +6 -0
  933. wbcore/signals/filters.py +3 -0
  934. wbcore/signals/instance_buttons.py +5 -0
  935. wbcore/signals/merge.py +4 -0
  936. wbcore/signals/models.py +8 -0
  937. wbcore/signals/permissions.py +3 -0
  938. wbcore/signals/serializers.py +5 -0
  939. wbcore/tasks.py +83 -0
  940. wbcore/templates/reversion/__init__.py +0 -0
  941. wbcore/templates/reversion/compare_detail.html +19 -0
  942. wbcore/test/__init__.py +30 -0
  943. wbcore/test/mixins.py +709 -0
  944. wbcore/test/signals.py +6 -0
  945. wbcore/test/tests.py +131 -0
  946. wbcore/test/utils.py +227 -0
  947. wbcore/tests/__init__.py +0 -0
  948. wbcore/tests/conftest.py +55 -0
  949. wbcore/tests/e2e/__init__.py +0 -0
  950. wbcore/tests/e2e/test_e2e.py +24 -0
  951. wbcore/tests/models.py +6 -0
  952. wbcore/tests/test_cache/__init__.py +0 -0
  953. wbcore/tests/test_cache/test_decorators.py +30 -0
  954. wbcore/tests/test_cache/test_mixins.py +29 -0
  955. wbcore/tests/test_cache/test_registry.py +57 -0
  956. wbcore/tests/test_configs.py +65 -0
  957. wbcore/tests/test_enums.py +54 -0
  958. wbcore/tests/test_fields/__init__.py +0 -0
  959. wbcore/tests/test_fields/test_boolean_fields.py +48 -0
  960. wbcore/tests/test_fields/test_choice_fields.py +48 -0
  961. wbcore/tests/test_fields/test_datetime_fields.py +151 -0
  962. wbcore/tests/test_fields/test_fields.py +23 -0
  963. wbcore/tests/test_fields/test_file_fields.py +52 -0
  964. wbcore/tests/test_fields/test_json_fields.py +28 -0
  965. wbcore/tests/test_fields/test_list_fields.py +27 -0
  966. wbcore/tests/test_fields/test_mixins.py +111 -0
  967. wbcore/tests/test_fields/test_number_fields.py +190 -0
  968. wbcore/tests/test_fields/test_other_fields.py +61 -0
  969. wbcore/tests/test_fields/test_primary_key_fields.py +52 -0
  970. wbcore/tests/test_fields/test_related.py +80 -0
  971. wbcore/tests/test_fields/test_text_fields.py +117 -0
  972. wbcore/tests/test_filters/__init__.py +0 -0
  973. wbcore/tests/test_filters/test_mixins.py +108 -0
  974. wbcore/tests/test_filters/test_pandas.py +114 -0
  975. wbcore/tests/test_list_display.py +31 -0
  976. wbcore/tests/test_models/__init__.py +0 -0
  977. wbcore/tests/test_models/test_fields.py +25 -0
  978. wbcore/tests/test_models/test_mixins.py +31 -0
  979. wbcore/tests/test_new_display/__init__.py +0 -0
  980. wbcore/tests/test_new_display/test_inlines.py +13 -0
  981. wbcore/tests/test_new_display/test_layouts.py +15 -0
  982. wbcore/tests/test_new_display/test_operators.py +15 -0
  983. wbcore/tests/test_new_display/test_pages.py +8 -0
  984. wbcore/tests/test_new_display/test_sections.py +0 -0
  985. wbcore/tests/test_new_display/test_shortcuts.py +38 -0
  986. wbcore/tests/test_new_display/test_utils.py +48 -0
  987. wbcore/tests/test_pagination.py +31 -0
  988. wbcore/tests/test_serializers/__init__.py +0 -0
  989. wbcore/tests/test_serializers/test_fields.py +140 -0
  990. wbcore/tests/test_serializers/test_mixins.py +53 -0
  991. wbcore/tests/test_serializers/test_related.py +77 -0
  992. wbcore/tests/test_something.py +39 -0
  993. wbcore/tests/test_utils/__init__.py +0 -0
  994. wbcore/tests/test_utils/test_date.py +49 -0
  995. wbcore/tests/test_utils/test_date_builder.py +99 -0
  996. wbcore/tests/test_utils/test_primary.py +79 -0
  997. wbcore/tests/test_utils/test_signals.py +38 -0
  998. wbcore/tests/test_viewsets.py +20 -0
  999. wbcore/urls.py +114 -0
  1000. wbcore/utils/__init__.py +13 -0
  1001. wbcore/utils/cache.py +8 -0
  1002. wbcore/utils/date.py +220 -0
  1003. wbcore/utils/date_builder/__init__.py +18 -0
  1004. wbcore/utils/date_builder/components.py +42 -0
  1005. wbcore/utils/date_builder/offsets.py +27 -0
  1006. wbcore/utils/deprecations.py +11 -0
  1007. wbcore/utils/enum.py +23 -0
  1008. wbcore/utils/figures.py +290 -0
  1009. wbcore/utils/html.py +8 -0
  1010. wbcore/utils/importlib.py +13 -0
  1011. wbcore/utils/itertools.py +40 -0
  1012. wbcore/utils/models.py +275 -0
  1013. wbcore/utils/numbers.py +69 -0
  1014. wbcore/utils/prettytable.py +35 -0
  1015. wbcore/utils/print.py +29 -0
  1016. wbcore/utils/renderers.py +13 -0
  1017. wbcore/utils/rrules.py +37 -0
  1018. wbcore/utils/serializers.py +8 -0
  1019. wbcore/utils/settings.py +5 -0
  1020. wbcore/utils/signals.py +36 -0
  1021. wbcore/utils/string_loader.py +42 -0
  1022. wbcore/utils/strings.py +77 -0
  1023. wbcore/utils/task.py +6 -0
  1024. wbcore/utils/urls.py +70 -0
  1025. wbcore/utils/views.py +201 -0
  1026. wbcore/views.py +26 -0
  1027. wbcore/viewsets/__init__.py +10 -0
  1028. wbcore/viewsets/encoders.py +18 -0
  1029. wbcore/viewsets/generics.py +5 -0
  1030. wbcore/viewsets/mixins.py +290 -0
  1031. wbcore/viewsets/utils.py +26 -0
  1032. wbcore/viewsets/viewsets.py +142 -0
  1033. wbcore-2.2.1.dist-info/METADATA +62 -0
  1034. wbcore-2.2.1.dist-info/RECORD +1035 -0
  1035. wbcore-2.2.1.dist-info/WHEEL +5 -0
@@ -0,0 +1,102 @@
1
+ import json
2
+ import os
3
+
4
+ import firebase_admin
5
+ from django.utils.html import strip_tags
6
+ from firebase_admin import messaging
7
+ from firebase_admin.credentials import Certificate
8
+ from wbcore.contrib.notifications.backends.abstract_backend import (
9
+ AbstractNotificationBackend,
10
+ )
11
+ from wbcore.contrib.notifications.models import Notification, NotificationUserToken
12
+ from wbcore.contrib.notifications.models.notification_types import (
13
+ NotificationTypeSetting,
14
+ )
15
+
16
+
17
+ class NotificationBackend(AbstractNotificationBackend):
18
+ FCM_OPTIONS = messaging.FCMOptions(analytics_label="notification")
19
+
20
+ @classmethod
21
+ def get_firebase_app(cls, certificate: Certificate):
22
+ try:
23
+ return firebase_admin.get_app()
24
+ except ValueError:
25
+ return firebase_admin.initialize_app(certificate)
26
+
27
+ @classmethod
28
+ def get_firebase_credentials(cls):
29
+ certificate = json.loads(os.environ.get("FIREBASE_SERVICE_WORKER_CREDENTIALS", "{}"))
30
+ return Certificate(certificate)
31
+
32
+ @classmethod
33
+ def send_notification(cls, notification: Notification):
34
+ app = cls.get_firebase_app(cls.get_firebase_credentials())
35
+ notification_user_settings = NotificationTypeSetting.objects.get(
36
+ notification_type=notification.notification_type,
37
+ user=notification.user,
38
+ )
39
+ tokens = NotificationUserToken.objects.filter_for_user_settings(notification_user_settings)
40
+ endpoint_data = {} # Firebase can't accept non-string value
41
+ if full_endpoint := notification.get_full_endpoint():
42
+ endpoint_data["endpoint"] = full_endpoint
43
+ expired_tokens = []
44
+ for token in tokens.filter(device_type=NotificationUserToken.NotificationDeviceType.MOBILE):
45
+ message = messaging.Message(
46
+ notification=messaging.Notification(
47
+ title=notification.title,
48
+ body=strip_tags(notification.body or ""),
49
+ ),
50
+ data=endpoint_data,
51
+ token=token.token,
52
+ android=messaging.AndroidConfig(
53
+ priority="high",
54
+ ttl=3600,
55
+ notification=messaging.AndroidNotification(
56
+ color="#f45342",
57
+ default_sound=True,
58
+ visibility="public",
59
+ default_vibrate_timings=True,
60
+ priority="max",
61
+ ),
62
+ ),
63
+ apns=messaging.APNSConfig(
64
+ payload=messaging.APNSPayload(
65
+ aps=messaging.Aps(sound=messaging.CriticalSound("default", volume=1.0))
66
+ ),
67
+ ),
68
+ fcm_options=cls.FCM_OPTIONS,
69
+ )
70
+ try:
71
+ messaging.send(message, False, app)
72
+ except messaging.UnregisteredError:
73
+ expired_tokens.append(token)
74
+
75
+ for token in tokens.filter(device_type=NotificationUserToken.NotificationDeviceType.WEB):
76
+ data = {
77
+ "title": notification.title,
78
+ "body": strip_tags(notification.body or ""),
79
+ "is_endpoint_internal": (
80
+ "true" if notification.is_endpoint_internal else "false"
81
+ ), # we need the data dictionary to contains only string values, otherwise firebase API complains
82
+ }
83
+ data.update(endpoint_data)
84
+ message = messaging.Message(
85
+ data=data,
86
+ token=token.token,
87
+ fcm_options=cls.FCM_OPTIONS,
88
+ )
89
+ try:
90
+ messaging.send(message, False, app)
91
+ except messaging.UnregisteredError:
92
+ expired_tokens.append(token)
93
+
94
+ for expired_token in expired_tokens:
95
+ expired_token.delete()
96
+
97
+ @classmethod
98
+ def get_configuration(cls) -> dict:
99
+ return {
100
+ "firebase_config": json.loads(os.environ.get("FIREBASE_WEB_CONFIG", "")),
101
+ "vapid_key": os.environ.get("FIREBASE_VAPID_KEY", ""),
102
+ }
@@ -0,0 +1,12 @@
1
+ from rest_framework.request import Request
2
+ from rest_framework.reverse import reverse
3
+ from wbcore.configs.decorators import register_config
4
+
5
+
6
+ @register_config
7
+ def notification_config(request: Request) -> tuple[str, dict[str, str]]:
8
+ return "notifications", {
9
+ "endpoint": reverse("wbcore:notifications:notification-list", request=request),
10
+ "token": reverse("wbcore:notifications:token", request=request),
11
+ "unread_notifications": reverse("wbcore:notifications:notification-unread-count", request=request),
12
+ }
@@ -0,0 +1,9 @@
1
+ from configurations import values
2
+ from django.db.models import options
3
+
4
+
5
+ class NotificationConfiguration:
6
+ options.DEFAULT_NAMES = options.DEFAULT_NAMES + ("notification_types",)
7
+ NOTIFICATION_BACKEND = values.Value(
8
+ "wbcore.contrib.notifications.backends.firebase.NotificationBackend", environ_prefix=None
9
+ )
@@ -0,0 +1,71 @@
1
+ from django.conf import settings
2
+ from django.contrib.auth import get_user_model
3
+ from django.db import transaction
4
+ from django.dispatch import receiver
5
+ from django.utils import timezone
6
+ from django.utils.module_loading import import_string
7
+ from django.utils.translation import gettext
8
+ from rest_framework.reverse import reverse
9
+ from wbcore.shares.signals import handle_widget_sharing
10
+
11
+ from .models import Notification, NotificationType, NotificationTypeSetting
12
+ from .tasks import send_notification_task
13
+
14
+ User = get_user_model()
15
+
16
+
17
+ def send_notification(
18
+ code: str,
19
+ title: str,
20
+ body: str,
21
+ user: User,
22
+ reverse_name: str | None = None,
23
+ reverse_args=None,
24
+ reverse_kwargs=None,
25
+ endpoint: str | None = None,
26
+ ):
27
+ """Method for dispatching a notification
28
+
29
+ Args:
30
+ code: The code pointing to a `NotificationType`
31
+ title: The title of the notification
32
+ body: The text of the notification
33
+ user: The user that should receive the notification
34
+ reverse_name: The reverse name of an endpoint attached to this notification
35
+ reverse_args: The arguments passed to the `reverse` function
36
+ reverse_kwargs: The keyword arguments passed to the `reverse` function
37
+ endpoint: The endpoint of resource. If provided, reverse_name + args + kwargs are disregarded
38
+ """
39
+ notification_user_settings = NotificationTypeSetting.objects.filter(
40
+ notification_type__code=code,
41
+ user=user,
42
+ )
43
+ if user.is_active and notification_user_settings.exists():
44
+ if not endpoint:
45
+ endpoint = reverse(reverse_name, reverse_args, reverse_kwargs) if reverse_name else None
46
+ notification = Notification.objects.create(
47
+ title=title,
48
+ body=body,
49
+ user=user,
50
+ notification_type=NotificationType.objects.get(code=code),
51
+ endpoint=endpoint,
52
+ sent=timezone.now(),
53
+ )
54
+ transaction.on_commit(lambda: send_notification_task.delay(notification.pk))
55
+
56
+
57
+ @receiver(handle_widget_sharing)
58
+ def share_notification(
59
+ request, widget_relative_endpoint, share=False, share_message=None, share_recipients=None, **kwargs
60
+ ):
61
+ if share and share_recipients and share_message:
62
+ for recipient in share_recipients:
63
+ send_notification(
64
+ code="workbench.system",
65
+ title=gettext("{} shared a widget with you").format(
66
+ import_string(settings.WBCORE_DEFAULT_USER_NAME)(request.user)
67
+ ),
68
+ user=recipient,
69
+ body=share_message,
70
+ endpoint=widget_relative_endpoint,
71
+ )
File without changes
@@ -0,0 +1,22 @@
1
+ import factory
2
+ from wbcore.contrib.notifications.models import (
3
+ NotificationType,
4
+ NotificationTypeSetting,
5
+ )
6
+
7
+
8
+ class NotificationTypeModelFactory(factory.django.DjangoModelFactory):
9
+ code = factory.Faker("pystr")
10
+ title = factory.Faker("pystr")
11
+ help_text = factory.Faker("pystr")
12
+
13
+ class Meta:
14
+ model = NotificationType
15
+
16
+
17
+ class NotificationTypeSettingModelFactory(factory.django.DjangoModelFactory):
18
+ notification_type = factory.SubFactory(NotificationTypeModelFactory)
19
+ user = factory.SubFactory("wbcore.contrib.authentication.factories.UserFactory")
20
+
21
+ class Meta:
22
+ model = NotificationTypeSetting
@@ -0,0 +1,16 @@
1
+ import factory
2
+ from wbcore.contrib.notifications.factories.notification_types import (
3
+ NotificationTypeModelFactory,
4
+ )
5
+ from wbcore.contrib.notifications.models import Notification
6
+
7
+
8
+ class NotificationModelFactory(factory.django.DjangoModelFactory):
9
+ title = factory.Faker("pystr")
10
+ body = factory.Faker("pystr")
11
+ notification_type = factory.SubFactory(NotificationTypeModelFactory)
12
+ user = factory.SubFactory("wbcore.contrib.authentication.factories.UserFactory")
13
+ endpoint = None
14
+
15
+ class Meta:
16
+ model = Notification
@@ -0,0 +1,11 @@
1
+ import factory
2
+ from wbcore.contrib.notifications.models.tokens import NotificationUserToken
3
+
4
+
5
+ class NotificationUserTokenModelFactory(factory.django.DjangoModelFactory):
6
+ user = factory.SubFactory("wbcore.contrib.authentication.factories.UserFactory")
7
+ token = factory.Faker("pystr")
8
+ device_type = factory.Iterator(NotificationUserToken.NotificationDeviceType.choices, getter=lambda c: c[0])
9
+
10
+ class Meta:
11
+ model = NotificationUserToken
@@ -0,0 +1,116 @@
1
+ # Generated by Django 4.1 on 2023-02-23 12:05
2
+
3
+ import django.db.models.deletion
4
+ from django.conf import settings
5
+ from django.db import migrations, models
6
+
7
+
8
+ class Migration(migrations.Migration):
9
+ initial = True
10
+
11
+ dependencies = [
12
+ migrations.swappable_dependency(settings.AUTH_USER_MODEL),
13
+ ("contenttypes", "0002_remove_content_type_name"),
14
+ ]
15
+
16
+ operations = [
17
+ migrations.CreateModel(
18
+ name="NotificationType",
19
+ fields=[
20
+ ("id", models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")),
21
+ ("code", models.CharField(max_length=128, unique=True)),
22
+ ("title", models.CharField(default="", max_length=128)),
23
+ ("help_text", models.CharField(default="", max_length=512)),
24
+ ("stale", models.BooleanField(default=False)),
25
+ ("default_enable_web", models.BooleanField(default=False)),
26
+ ("default_enable_mobile", models.BooleanField(default=False)),
27
+ ("default_enable_email", models.BooleanField(default=False)),
28
+ (
29
+ "contenttype",
30
+ models.ForeignKey(
31
+ blank=True,
32
+ null=True,
33
+ on_delete=django.db.models.deletion.SET_NULL,
34
+ related_name="+",
35
+ to="contenttypes.contenttype",
36
+ ),
37
+ ),
38
+ ],
39
+ options={
40
+ "verbose_name": "Notification Type",
41
+ "verbose_name_plural": "Notification Types",
42
+ },
43
+ ),
44
+ migrations.CreateModel(
45
+ name="NotificationUserToken",
46
+ fields=[
47
+ ("id", models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")),
48
+ ("token", models.CharField(max_length=256)),
49
+ ("device_type", models.CharField(choices=[("WEB", "Web"), ("MOBILE", "Mobile")], max_length=16)),
50
+ (
51
+ "user",
52
+ models.ForeignKey(
53
+ on_delete=django.db.models.deletion.CASCADE,
54
+ related_name="notifications_tokens",
55
+ to=settings.AUTH_USER_MODEL,
56
+ ),
57
+ ),
58
+ ],
59
+ ),
60
+ migrations.CreateModel(
61
+ name="NotificationTypeSetting",
62
+ fields=[
63
+ ("id", models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")),
64
+ ("enable_web", models.BooleanField(default=False)),
65
+ ("enable_mobile", models.BooleanField(default=False)),
66
+ ("enable_email", models.BooleanField(default=False)),
67
+ (
68
+ "notification_type",
69
+ models.ForeignKey(
70
+ on_delete=django.db.models.deletion.CASCADE,
71
+ related_name="user_settings",
72
+ to="notifications.notificationtype",
73
+ ),
74
+ ),
75
+ (
76
+ "user",
77
+ models.ForeignKey(
78
+ on_delete=django.db.models.deletion.CASCADE,
79
+ related_name="wbnotification_user_settings",
80
+ to=settings.AUTH_USER_MODEL,
81
+ ),
82
+ ),
83
+ ],
84
+ ),
85
+ migrations.CreateModel(
86
+ name="Notification",
87
+ fields=[
88
+ ("id", models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")),
89
+ ("title", models.CharField(max_length=255)),
90
+ ("body", models.TextField(default="")),
91
+ ("endpoint", models.CharField(blank=True, max_length=255, null=True)),
92
+ ("sent", models.DateTimeField(blank=True, null=True)),
93
+ ("read", models.DateTimeField(blank=True, null=True)),
94
+ (
95
+ "notification_type",
96
+ models.ForeignKey(
97
+ on_delete=django.db.models.deletion.CASCADE,
98
+ related_name="notifications",
99
+ to="notifications.notificationtype",
100
+ ),
101
+ ),
102
+ (
103
+ "user",
104
+ models.ForeignKey(
105
+ on_delete=django.db.models.deletion.CASCADE,
106
+ related_name="notifications_notifications",
107
+ to=settings.AUTH_USER_MODEL,
108
+ ),
109
+ ),
110
+ ],
111
+ options={
112
+ "verbose_name": "Notification",
113
+ "verbose_name_plural": "Notifications",
114
+ },
115
+ ),
116
+ ]
@@ -0,0 +1,18 @@
1
+ # Generated by Django 4.1 on 2023-03-30 09:21
2
+
3
+ from django.db import migrations, models
4
+
5
+
6
+ class Migration(migrations.Migration):
7
+ dependencies = [
8
+ ("notifications", "0001_initial"),
9
+ ]
10
+
11
+ operations = [
12
+ migrations.AddConstraint(
13
+ model_name="notificationusertoken",
14
+ constraint=models.UniqueConstraint(
15
+ fields=("user", "token", "device_type"), name="unique_user_token_device"
16
+ ),
17
+ ),
18
+ ]
@@ -0,0 +1,17 @@
1
+ # Generated by Django 4.1.9 on 2023-05-24 08:39
2
+
3
+ from django.db import migrations, models
4
+
5
+
6
+ class Migration(migrations.Migration):
7
+ dependencies = [
8
+ ("notifications", "0002_notificationusertoken_unique_user_token_device"),
9
+ ]
10
+
11
+ operations = [
12
+ migrations.AddField(
13
+ model_name="notificationusertoken",
14
+ name="updated",
15
+ field=models.DateTimeField(auto_now=True),
16
+ ),
17
+ ]
@@ -0,0 +1,17 @@
1
+ # Generated by Django 4.1.9 on 2023-05-30 07:44
2
+
3
+ from django.db import migrations, models
4
+
5
+
6
+ class Migration(migrations.Migration):
7
+ dependencies = [
8
+ ("notifications", "0003_notificationusertoken_updated"),
9
+ ]
10
+
11
+ operations = [
12
+ migrations.AlterField(
13
+ model_name="notification",
14
+ name="body",
15
+ field=models.TextField(blank=True, null=True),
16
+ ),
17
+ ]
@@ -0,0 +1,17 @@
1
+ # Generated by Django 5.0.3 on 2024-04-24 11:28
2
+
3
+ from django.db import migrations, models
4
+
5
+
6
+ class Migration(migrations.Migration):
7
+ dependencies = [
8
+ ("notifications", "0004_alter_notification_body"),
9
+ ]
10
+
11
+ operations = [
12
+ migrations.AlterField(
13
+ model_name="notification",
14
+ name="endpoint",
15
+ field=models.CharField(blank=True, max_length=2048, null=True),
16
+ ),
17
+ ]
@@ -0,0 +1,25 @@
1
+ # Generated by Django 5.0.9 on 2024-12-05 13:27
2
+
3
+ import django.utils.timezone
4
+ from django.db import migrations, models
5
+
6
+
7
+ def set_created(apps, schema_editor):
8
+ Notification = apps.get_model("notifications", "Notification")
9
+ Notification.objects.filter(sent__isnull=False).update(created=models.F("sent"))
10
+
11
+
12
+ class Migration(migrations.Migration):
13
+ dependencies = [
14
+ ("notifications", "0005_alter_notification_endpoint"),
15
+ ]
16
+
17
+ operations = [
18
+ migrations.AddField(
19
+ model_name="notification",
20
+ name="created",
21
+ field=models.DateTimeField(auto_now_add=True, default=django.utils.timezone.now),
22
+ preserve_default=False,
23
+ ),
24
+ migrations.RunPython(set_created),
25
+ ]
File without changes
@@ -0,0 +1,3 @@
1
+ from .notification_types import NotificationType, NotificationTypeSetting
2
+ from .notifications import Notification
3
+ from .tokens import NotificationUserToken
@@ -0,0 +1,112 @@
1
+ from contextlib import suppress
2
+
3
+ from django.contrib.auth import get_user_model
4
+ from django.db import models
5
+ from django.db.models.signals import post_save
6
+ from django.dispatch import receiver
7
+ from guardian.utils import get_anonymous_user
8
+ from wbcore.models import WBModel
9
+
10
+
11
+ class NotificationType(WBModel):
12
+ code = models.CharField(max_length=128, unique=True)
13
+ title = models.CharField(max_length=128, default="")
14
+ help_text = models.CharField(max_length=512, default="")
15
+
16
+ stale = models.BooleanField(default=False)
17
+ contenttype = models.ForeignKey(
18
+ to="contenttypes.ContentType", related_name="+", null=True, blank=True, on_delete=models.SET_NULL
19
+ )
20
+
21
+ default_enable_web = models.BooleanField(default=False)
22
+ default_enable_mobile = models.BooleanField(default=False)
23
+ default_enable_email = models.BooleanField(default=False)
24
+
25
+ def __str__(self) -> str:
26
+ return f"{self.title}"
27
+
28
+ class Meta:
29
+ verbose_name = "Notification Type"
30
+ verbose_name_plural = "Notification Types"
31
+
32
+ @classmethod
33
+ def get_endpoint_basename(cls):
34
+ return "wbcore:notifications:notification_type"
35
+
36
+ @classmethod
37
+ def get_representation_endpoint(cls):
38
+ return "wbcore:notifications:notification_type_representation-list"
39
+
40
+ @classmethod
41
+ def get_representation_value_key(cls):
42
+ return "id"
43
+
44
+ @classmethod
45
+ def get_representation_label_key(cls):
46
+ return "{{title}}"
47
+
48
+
49
+ class NotificationTypeSetting(WBModel):
50
+ notification_type = models.ForeignKey(
51
+ to="notifications.NotificationType", related_name="user_settings", on_delete=models.CASCADE
52
+ )
53
+ user = models.ForeignKey(
54
+ to=get_user_model(), related_name="wbnotification_user_settings", on_delete=models.CASCADE
55
+ )
56
+
57
+ enable_web = models.BooleanField(default=False)
58
+ enable_mobile = models.BooleanField(default=False)
59
+ enable_email = models.BooleanField(default=False)
60
+
61
+ def __str__(self) -> str:
62
+ return f"{self.user}: {self.notification_type} ({self.enable_web}/{self.enable_mobile}/{self.enable_email})"
63
+
64
+ @classmethod
65
+ def get_endpoint_basename(cls):
66
+ return "wbcore:notifications:notification_type_setting"
67
+
68
+ @classmethod
69
+ def get_representation_endpoint(cls):
70
+ return "wbcore:notifications:notification_type_setting_representation-list"
71
+
72
+ @classmethod
73
+ def get_representation_value_key(cls):
74
+ return "id"
75
+
76
+ @classmethod
77
+ def get_representation_label_key(cls):
78
+ return "{{notification_type}}"
79
+
80
+
81
+ @receiver(post_save, sender="notifications.NotificationType")
82
+ def post_save_notification_type(instance, **kwargs):
83
+ anonymous_user = get_anonymous_user()
84
+ for user in get_user_model().objects.filter(~models.Q(pk=anonymous_user.pk)):
85
+ NotificationTypeSetting.objects.get_or_create(
86
+ notification_type=instance,
87
+ user=user,
88
+ defaults={
89
+ "enable_web": instance.default_enable_web,
90
+ "enable_mobile": instance.default_enable_mobile,
91
+ "enable_email": instance.default_enable_email,
92
+ },
93
+ )
94
+
95
+
96
+ @receiver(post_save, sender=get_user_model())
97
+ def post_save_user(instance, **kwargs):
98
+ with suppress(get_user_model().DoesNotExist):
99
+ anonymous_user = get_anonymous_user()
100
+ if instance.pk == anonymous_user.pk:
101
+ return
102
+
103
+ for notification_type in NotificationType.objects.all():
104
+ NotificationTypeSetting.objects.get_or_create(
105
+ notification_type=notification_type,
106
+ user=instance,
107
+ defaults={
108
+ "enable_web": notification_type.default_enable_web,
109
+ "enable_mobile": notification_type.default_enable_mobile,
110
+ "enable_email": notification_type.default_enable_email,
111
+ },
112
+ )
@@ -0,0 +1,81 @@
1
+ import urllib
2
+ from contextlib import suppress
3
+
4
+ from django.contrib.auth import get_user_model
5
+ from django.core.validators import URLValidator, ValidationError
6
+ from django.db import models
7
+ from django.urls import Resolver404, resolve
8
+ from django.utils.functional import cached_property
9
+ from wbcore.contrib.notifications.utils import base_domain
10
+
11
+
12
+ class Notification(models.Model):
13
+ title = models.CharField(max_length=255)
14
+ body = models.TextField(null=True, blank=True)
15
+
16
+ user = models.ForeignKey(to=get_user_model(), related_name="notifications_notifications", on_delete=models.CASCADE)
17
+ notification_type = models.ForeignKey(
18
+ to="notifications.NotificationType", related_name="notifications", on_delete=models.CASCADE
19
+ )
20
+ endpoint = models.CharField(max_length=2048, null=True, blank=True)
21
+
22
+ created = models.DateTimeField(auto_now_add=True)
23
+ sent = models.DateTimeField(null=True, blank=True)
24
+ read = models.DateTimeField(null=True, blank=True)
25
+
26
+ def __str__(self) -> str:
27
+ return f"{self.user} {self.title}"
28
+
29
+ class Meta:
30
+ verbose_name = "Notification"
31
+ verbose_name_plural = "Notifications"
32
+
33
+ notification_types = [
34
+ (
35
+ "workbench.system",
36
+ "System Notifications",
37
+ "System Notifications.",
38
+ True,
39
+ True,
40
+ False,
41
+ ),
42
+ ]
43
+
44
+ @cached_property
45
+ def is_endpoint_internal(self) -> bool:
46
+ with suppress(Resolver404):
47
+ if self.endpoint:
48
+ resolve(
49
+ urllib.parse.urlsplit(urllib.parse.unquote(self.endpoint)).path
50
+ ) # we need to truncate query parameters
51
+ return True
52
+ return False
53
+
54
+ @cached_property
55
+ def is_endpoint_valid(self) -> bool:
56
+ try:
57
+ URLValidator()(self.endpoint)
58
+ return True
59
+ except ValidationError:
60
+ return False
61
+
62
+ def get_full_endpoint(self, as_shareable_internal_link: bool = False) -> str | None:
63
+ if self.is_endpoint_internal:
64
+ if as_shareable_internal_link:
65
+ return f"{base_domain()}?widget_endpoint={self.endpoint}"
66
+ else:
67
+ return f"{base_domain()}{self.endpoint}"
68
+ elif self.is_endpoint_valid:
69
+ return self.endpoint
70
+
71
+ @classmethod
72
+ def get_endpoint_basename(cls) -> str:
73
+ return "wbcore:notifications:notification"
74
+
75
+ @classmethod
76
+ def get_representation_value_key(cls) -> str:
77
+ return "id"
78
+
79
+ @classmethod
80
+ def get_representation_label_key(cls) -> str:
81
+ return "{{title}}"