wbfdm 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.

Potentially problematic release.


This version of wbfdm might be problematic. Click here for more details.

Files changed (337) hide show
  1. wbfdm/__init__.py +2 -0
  2. wbfdm/admin/__init__.py +42 -0
  3. wbfdm/admin/classifications.py +39 -0
  4. wbfdm/admin/esg.py +23 -0
  5. wbfdm/admin/exchanges.py +53 -0
  6. wbfdm/admin/instrument_lists.py +23 -0
  7. wbfdm/admin/instrument_prices.py +62 -0
  8. wbfdm/admin/instrument_requests.py +33 -0
  9. wbfdm/admin/instruments.py +117 -0
  10. wbfdm/admin/instruments_relationships.py +25 -0
  11. wbfdm/admin/options.py +101 -0
  12. wbfdm/analysis/__init__.py +2 -0
  13. wbfdm/analysis/esg/__init__.py +0 -0
  14. wbfdm/analysis/esg/enums.py +82 -0
  15. wbfdm/analysis/esg/esg_analysis.py +217 -0
  16. wbfdm/analysis/esg/utils.py +13 -0
  17. wbfdm/analysis/financial_analysis/__init__.py +1 -0
  18. wbfdm/analysis/financial_analysis/financial_metric_analysis.py +88 -0
  19. wbfdm/analysis/financial_analysis/financial_ratio_analysis.py +125 -0
  20. wbfdm/analysis/financial_analysis/financial_statistics_analysis.py +271 -0
  21. wbfdm/analysis/financial_analysis/statement_with_estimates.py +558 -0
  22. wbfdm/analysis/financial_analysis/utils.py +316 -0
  23. wbfdm/analysis/technical_analysis/__init__.py +1 -0
  24. wbfdm/analysis/technical_analysis/technical_analysis.py +138 -0
  25. wbfdm/analysis/technical_analysis/traces.py +165 -0
  26. wbfdm/analysis/utils.py +32 -0
  27. wbfdm/apps.py +14 -0
  28. wbfdm/contrib/__init__.py +0 -0
  29. wbfdm/contrib/dsws/__init__.py +0 -0
  30. wbfdm/contrib/dsws/client.py +285 -0
  31. wbfdm/contrib/internal/__init__.py +0 -0
  32. wbfdm/contrib/internal/dataloaders/__init__.py +0 -0
  33. wbfdm/contrib/internal/dataloaders/market_data.py +87 -0
  34. wbfdm/contrib/metric/__init__.py +0 -0
  35. wbfdm/contrib/metric/admin/__init__.py +2 -0
  36. wbfdm/contrib/metric/admin/instruments.py +12 -0
  37. wbfdm/contrib/metric/admin/metrics.py +43 -0
  38. wbfdm/contrib/metric/apps.py +10 -0
  39. wbfdm/contrib/metric/backends/__init__.py +2 -0
  40. wbfdm/contrib/metric/backends/base.py +159 -0
  41. wbfdm/contrib/metric/backends/performances.py +265 -0
  42. wbfdm/contrib/metric/backends/statistics.py +182 -0
  43. wbfdm/contrib/metric/decorators.py +14 -0
  44. wbfdm/contrib/metric/dispatch.py +23 -0
  45. wbfdm/contrib/metric/dto.py +88 -0
  46. wbfdm/contrib/metric/exceptions.py +6 -0
  47. wbfdm/contrib/metric/factories.py +33 -0
  48. wbfdm/contrib/metric/filters.py +28 -0
  49. wbfdm/contrib/metric/migrations/0001_initial.py +88 -0
  50. wbfdm/contrib/metric/migrations/0002_remove_instrumentmetric_unique_instrument_metric_and_more.py +26 -0
  51. wbfdm/contrib/metric/migrations/__init__.py +0 -0
  52. wbfdm/contrib/metric/models.py +180 -0
  53. wbfdm/contrib/metric/orchestrators.py +94 -0
  54. wbfdm/contrib/metric/registry.py +80 -0
  55. wbfdm/contrib/metric/serializers.py +44 -0
  56. wbfdm/contrib/metric/tasks.py +27 -0
  57. wbfdm/contrib/metric/tests/__init__.py +0 -0
  58. wbfdm/contrib/metric/tests/backends/__init__.py +0 -0
  59. wbfdm/contrib/metric/tests/backends/test_performances.py +152 -0
  60. wbfdm/contrib/metric/tests/backends/test_statistics.py +48 -0
  61. wbfdm/contrib/metric/tests/conftest.py +92 -0
  62. wbfdm/contrib/metric/tests/test_dto.py +73 -0
  63. wbfdm/contrib/metric/tests/test_models.py +72 -0
  64. wbfdm/contrib/metric/tests/test_tasks.py +24 -0
  65. wbfdm/contrib/metric/tests/test_viewsets.py +79 -0
  66. wbfdm/contrib/metric/urls.py +19 -0
  67. wbfdm/contrib/metric/viewsets/__init__.py +1 -0
  68. wbfdm/contrib/metric/viewsets/configs/__init__.py +1 -0
  69. wbfdm/contrib/metric/viewsets/configs/display.py +92 -0
  70. wbfdm/contrib/metric/viewsets/configs/menus.py +11 -0
  71. wbfdm/contrib/metric/viewsets/configs/utils.py +137 -0
  72. wbfdm/contrib/metric/viewsets/mixins.py +245 -0
  73. wbfdm/contrib/metric/viewsets/viewsets.py +40 -0
  74. wbfdm/contrib/msci/__init__.py +0 -0
  75. wbfdm/contrib/msci/client.py +92 -0
  76. wbfdm/contrib/msci/dataloaders/__init__.py +0 -0
  77. wbfdm/contrib/msci/dataloaders/esg.py +87 -0
  78. wbfdm/contrib/msci/dataloaders/esg_controversies.py +81 -0
  79. wbfdm/contrib/msci/sync.py +58 -0
  80. wbfdm/contrib/msci/tests/__init__.py +0 -0
  81. wbfdm/contrib/msci/tests/conftest.py +1 -0
  82. wbfdm/contrib/msci/tests/test_client.py +70 -0
  83. wbfdm/contrib/qa/__init__.py +0 -0
  84. wbfdm/contrib/qa/apps.py +22 -0
  85. wbfdm/contrib/qa/database_routers.py +25 -0
  86. wbfdm/contrib/qa/dataloaders/__init__.py +0 -0
  87. wbfdm/contrib/qa/dataloaders/adjustments.py +56 -0
  88. wbfdm/contrib/qa/dataloaders/corporate_actions.py +59 -0
  89. wbfdm/contrib/qa/dataloaders/financials.py +83 -0
  90. wbfdm/contrib/qa/dataloaders/market_data.py +117 -0
  91. wbfdm/contrib/qa/dataloaders/officers.py +59 -0
  92. wbfdm/contrib/qa/dataloaders/reporting_dates.py +67 -0
  93. wbfdm/contrib/qa/dataloaders/statements.py +267 -0
  94. wbfdm/contrib/qa/tasks.py +0 -0
  95. wbfdm/dataloaders/__init__.py +0 -0
  96. wbfdm/dataloaders/cache.py +129 -0
  97. wbfdm/dataloaders/protocols.py +112 -0
  98. wbfdm/dataloaders/proxies.py +201 -0
  99. wbfdm/dataloaders/types.py +209 -0
  100. wbfdm/dynamic_preferences_registry.py +45 -0
  101. wbfdm/enums.py +657 -0
  102. wbfdm/factories/__init__.py +13 -0
  103. wbfdm/factories/classifications.py +56 -0
  104. wbfdm/factories/controversies.py +27 -0
  105. wbfdm/factories/exchanges.py +21 -0
  106. wbfdm/factories/instrument_list.py +22 -0
  107. wbfdm/factories/instrument_prices.py +79 -0
  108. wbfdm/factories/instruments.py +63 -0
  109. wbfdm/factories/instruments_relationships.py +31 -0
  110. wbfdm/factories/options.py +66 -0
  111. wbfdm/figures/__init__.py +1 -0
  112. wbfdm/figures/financials/__init__.py +1 -0
  113. wbfdm/figures/financials/financial_analysis_charts.py +469 -0
  114. wbfdm/figures/financials/financials_charts.py +711 -0
  115. wbfdm/filters/__init__.py +31 -0
  116. wbfdm/filters/classifications.py +100 -0
  117. wbfdm/filters/exchanges.py +22 -0
  118. wbfdm/filters/financials.py +95 -0
  119. wbfdm/filters/financials_analysis.py +119 -0
  120. wbfdm/filters/instrument_prices.py +112 -0
  121. wbfdm/filters/instruments.py +198 -0
  122. wbfdm/filters/utils.py +44 -0
  123. wbfdm/import_export/__init__.py +0 -0
  124. wbfdm/import_export/backends/__init__.py +0 -0
  125. wbfdm/import_export/backends/cbinsights/__init__.py +2 -0
  126. wbfdm/import_export/backends/cbinsights/deals.py +44 -0
  127. wbfdm/import_export/backends/cbinsights/equities.py +41 -0
  128. wbfdm/import_export/backends/cbinsights/mixin.py +15 -0
  129. wbfdm/import_export/backends/cbinsights/utils/__init__.py +0 -0
  130. wbfdm/import_export/backends/cbinsights/utils/classifications.py +4150 -0
  131. wbfdm/import_export/backends/cbinsights/utils/client.py +217 -0
  132. wbfdm/import_export/backends/refinitiv/__init__.py +5 -0
  133. wbfdm/import_export/backends/refinitiv/daily_fundamental.py +36 -0
  134. wbfdm/import_export/backends/refinitiv/fiscal_period.py +63 -0
  135. wbfdm/import_export/backends/refinitiv/forecast.py +178 -0
  136. wbfdm/import_export/backends/refinitiv/fundamental.py +103 -0
  137. wbfdm/import_export/backends/refinitiv/geographic_segment.py +32 -0
  138. wbfdm/import_export/backends/refinitiv/instrument.py +55 -0
  139. wbfdm/import_export/backends/refinitiv/instrument_price.py +77 -0
  140. wbfdm/import_export/backends/refinitiv/mixin.py +29 -0
  141. wbfdm/import_export/backends/refinitiv/utils/__init__.py +1 -0
  142. wbfdm/import_export/backends/refinitiv/utils/controller.py +182 -0
  143. wbfdm/import_export/handlers/__init__.py +0 -0
  144. wbfdm/import_export/handlers/instrument.py +253 -0
  145. wbfdm/import_export/handlers/instrument_list.py +101 -0
  146. wbfdm/import_export/handlers/instrument_price.py +71 -0
  147. wbfdm/import_export/handlers/option.py +54 -0
  148. wbfdm/import_export/handlers/private_equities.py +49 -0
  149. wbfdm/import_export/parsers/__init__.py +0 -0
  150. wbfdm/import_export/parsers/cbinsights/__init__.py +0 -0
  151. wbfdm/import_export/parsers/cbinsights/deals.py +39 -0
  152. wbfdm/import_export/parsers/cbinsights/equities.py +56 -0
  153. wbfdm/import_export/parsers/cbinsights/fundamentals.py +45 -0
  154. wbfdm/import_export/parsers/refinitiv/__init__.py +0 -0
  155. wbfdm/import_export/parsers/refinitiv/daily_fundamental.py +7 -0
  156. wbfdm/import_export/parsers/refinitiv/forecast.py +7 -0
  157. wbfdm/import_export/parsers/refinitiv/fundamental.py +9 -0
  158. wbfdm/import_export/parsers/refinitiv/geographic_segment.py +7 -0
  159. wbfdm/import_export/parsers/refinitiv/instrument.py +75 -0
  160. wbfdm/import_export/parsers/refinitiv/instrument_price.py +26 -0
  161. wbfdm/import_export/parsers/refinitiv/utils.py +96 -0
  162. wbfdm/import_export/resources/__init__.py +0 -0
  163. wbfdm/import_export/resources/classification.py +23 -0
  164. wbfdm/import_export/resources/instrument_prices.py +33 -0
  165. wbfdm/import_export/resources/instruments.py +176 -0
  166. wbfdm/jinja2.py +7 -0
  167. wbfdm/management/__init__.py +30 -0
  168. wbfdm/menu.py +11 -0
  169. wbfdm/migrations/0001_initial.py +71 -0
  170. wbfdm/migrations/0002_rename_statements_instrumentlookup_financials_and_more.py +144 -0
  171. wbfdm/migrations/0003_instrument_estimate_backend_and_more.py +34 -0
  172. wbfdm/migrations/0004_rename_financials_instrumentlookup_statements_and_more.py +86 -0
  173. wbfdm/migrations/0005_instrument_corporate_action_backend.py +29 -0
  174. wbfdm/migrations/0006_instrument_officer_backend.py +29 -0
  175. wbfdm/migrations/0007_instrument_country_instrument_currency_and_more.py +117 -0
  176. wbfdm/migrations/0008_controversy.py +75 -0
  177. wbfdm/migrations/0009_alter_controversy_flag_alter_controversy_initiated_and_more.py +85 -0
  178. wbfdm/migrations/0010_classification_classificationgroup_deal_exchange_and_more.py +1299 -0
  179. wbfdm/migrations/0011_delete_instrumentlookup_instrument_corporate_actions_and_more.py +169 -0
  180. wbfdm/migrations/0012_instrumentprice_created_instrumentprice_modified.py +564 -0
  181. wbfdm/migrations/0013_instrument_is_investable_universe_and_more.py +199 -0
  182. wbfdm/migrations/0014_alter_controversy_instrument.py +22 -0
  183. wbfdm/migrations/0015_instrument_instrument_investible_index.py +16 -0
  184. wbfdm/migrations/0016_instrumenttype_name_repr.py +18 -0
  185. wbfdm/migrations/0017_instrument_instrument_security_index.py +16 -0
  186. wbfdm/migrations/0018_instrument_instrument_level_index.py +20 -0
  187. wbfdm/migrations/0019_alter_controversy_source.py +17 -0
  188. wbfdm/migrations/0020_optionaggregate_option_and_more.py +249 -0
  189. wbfdm/migrations/0021_delete_instrumentdailystatistics.py +15 -0
  190. wbfdm/migrations/0022_instrument_cusip_option_open_interest_20d_and_more.py +91 -0
  191. wbfdm/migrations/0023_instrument_unique_ric_instrument_unique_rmc_and_more.py +53 -0
  192. wbfdm/migrations/0024_option_open_interest_10d_option_volume_10d_and_more.py +36 -0
  193. wbfdm/migrations/0025_instrument_is_primary_and_more.py +29 -0
  194. wbfdm/migrations/0026_instrument_is_cash_equivalent.py +30 -0
  195. wbfdm/migrations/0027_remove_instrument_unique_ric_and_more.py +100 -0
  196. wbfdm/migrations/__init__.py +0 -0
  197. wbfdm/models/__init__.py +4 -0
  198. wbfdm/models/esg/__init__.py +1 -0
  199. wbfdm/models/esg/controversies.py +81 -0
  200. wbfdm/models/exchanges/__init__.py +1 -0
  201. wbfdm/models/exchanges/exchanges.py +223 -0
  202. wbfdm/models/fields.py +117 -0
  203. wbfdm/models/fk_fields.py +403 -0
  204. wbfdm/models/indicators.py +0 -0
  205. wbfdm/models/instruments/__init__.py +19 -0
  206. wbfdm/models/instruments/classifications.py +265 -0
  207. wbfdm/models/instruments/instrument_lists.py +120 -0
  208. wbfdm/models/instruments/instrument_prices.py +540 -0
  209. wbfdm/models/instruments/instrument_relationships.py +251 -0
  210. wbfdm/models/instruments/instrument_requests.py +196 -0
  211. wbfdm/models/instruments/instruments.py +991 -0
  212. wbfdm/models/instruments/llm/__init__.py +1 -0
  213. wbfdm/models/instruments/llm/create_instrument_news_relationships.py +78 -0
  214. wbfdm/models/instruments/mixin/__init__.py +0 -0
  215. wbfdm/models/instruments/mixin/financials_computed.py +804 -0
  216. wbfdm/models/instruments/mixin/financials_serializer_fields.py +1407 -0
  217. wbfdm/models/instruments/mixin/instruments.py +294 -0
  218. wbfdm/models/instruments/options.py +225 -0
  219. wbfdm/models/instruments/private_equities.py +59 -0
  220. wbfdm/models/instruments/querysets.py +73 -0
  221. wbfdm/models/instruments/utils.py +41 -0
  222. wbfdm/preferences.py +21 -0
  223. wbfdm/serializers/__init__.py +4 -0
  224. wbfdm/serializers/esg.py +36 -0
  225. wbfdm/serializers/exchanges.py +39 -0
  226. wbfdm/serializers/instruments/__init__.py +37 -0
  227. wbfdm/serializers/instruments/classifications.py +139 -0
  228. wbfdm/serializers/instruments/instrument_lists.py +61 -0
  229. wbfdm/serializers/instruments/instrument_prices.py +73 -0
  230. wbfdm/serializers/instruments/instrument_relationships.py +170 -0
  231. wbfdm/serializers/instruments/instrument_requests.py +61 -0
  232. wbfdm/serializers/instruments/instruments.py +274 -0
  233. wbfdm/serializers/instruments/mixins.py +104 -0
  234. wbfdm/serializers/officers.py +20 -0
  235. wbfdm/signals.py +7 -0
  236. wbfdm/sync/__init__.py +0 -0
  237. wbfdm/sync/abstract.py +31 -0
  238. wbfdm/sync/runner.py +22 -0
  239. wbfdm/tasks.py +69 -0
  240. wbfdm/tests/__init__.py +0 -0
  241. wbfdm/tests/analysis/__init__.py +0 -0
  242. wbfdm/tests/analysis/financial_analysis/__init__.py +0 -0
  243. wbfdm/tests/analysis/financial_analysis/test_statement_with_estimates.py +392 -0
  244. wbfdm/tests/analysis/financial_analysis/test_utils.py +322 -0
  245. wbfdm/tests/analysis/test_esg.py +159 -0
  246. wbfdm/tests/conftest.py +92 -0
  247. wbfdm/tests/dataloaders/__init__.py +0 -0
  248. wbfdm/tests/dataloaders/test_cache.py +73 -0
  249. wbfdm/tests/models/__init__.py +0 -0
  250. wbfdm/tests/models/test_classifications.py +99 -0
  251. wbfdm/tests/models/test_exchanges.py +7 -0
  252. wbfdm/tests/models/test_instrument_list.py +117 -0
  253. wbfdm/tests/models/test_instrument_prices.py +306 -0
  254. wbfdm/tests/models/test_instruments.py +202 -0
  255. wbfdm/tests/models/test_merge.py +99 -0
  256. wbfdm/tests/models/test_options.py +69 -0
  257. wbfdm/tests/test_tasks.py +6 -0
  258. wbfdm/tests/tests.py +10 -0
  259. wbfdm/urls.py +222 -0
  260. wbfdm/utils.py +54 -0
  261. wbfdm/viewsets/__init__.py +10 -0
  262. wbfdm/viewsets/configs/__init__.py +5 -0
  263. wbfdm/viewsets/configs/buttons/__init__.py +8 -0
  264. wbfdm/viewsets/configs/buttons/classifications.py +23 -0
  265. wbfdm/viewsets/configs/buttons/exchanges.py +9 -0
  266. wbfdm/viewsets/configs/buttons/instrument_prices.py +49 -0
  267. wbfdm/viewsets/configs/buttons/instruments.py +283 -0
  268. wbfdm/viewsets/configs/display/__init__.py +22 -0
  269. wbfdm/viewsets/configs/display/classifications.py +138 -0
  270. wbfdm/viewsets/configs/display/esg.py +75 -0
  271. wbfdm/viewsets/configs/display/exchanges.py +42 -0
  272. wbfdm/viewsets/configs/display/instrument_lists.py +137 -0
  273. wbfdm/viewsets/configs/display/instrument_prices.py +199 -0
  274. wbfdm/viewsets/configs/display/instrument_requests.py +116 -0
  275. wbfdm/viewsets/configs/display/instruments.py +618 -0
  276. wbfdm/viewsets/configs/display/instruments_relationships.py +65 -0
  277. wbfdm/viewsets/configs/display/monthly_performances.py +72 -0
  278. wbfdm/viewsets/configs/display/officers.py +16 -0
  279. wbfdm/viewsets/configs/display/prices.py +21 -0
  280. wbfdm/viewsets/configs/display/statement_with_estimates.py +101 -0
  281. wbfdm/viewsets/configs/display/statements.py +48 -0
  282. wbfdm/viewsets/configs/endpoints/__init__.py +41 -0
  283. wbfdm/viewsets/configs/endpoints/classifications.py +87 -0
  284. wbfdm/viewsets/configs/endpoints/esg.py +20 -0
  285. wbfdm/viewsets/configs/endpoints/exchanges.py +6 -0
  286. wbfdm/viewsets/configs/endpoints/financials_analysis.py +65 -0
  287. wbfdm/viewsets/configs/endpoints/instrument_lists.py +38 -0
  288. wbfdm/viewsets/configs/endpoints/instrument_prices.py +51 -0
  289. wbfdm/viewsets/configs/endpoints/instrument_requests.py +20 -0
  290. wbfdm/viewsets/configs/endpoints/instruments.py +13 -0
  291. wbfdm/viewsets/configs/endpoints/instruments_relationships.py +31 -0
  292. wbfdm/viewsets/configs/endpoints/statements.py +6 -0
  293. wbfdm/viewsets/configs/menus/__init__.py +9 -0
  294. wbfdm/viewsets/configs/menus/classifications.py +19 -0
  295. wbfdm/viewsets/configs/menus/exchanges.py +10 -0
  296. wbfdm/viewsets/configs/menus/instrument_lists.py +10 -0
  297. wbfdm/viewsets/configs/menus/instruments.py +20 -0
  298. wbfdm/viewsets/configs/menus/instruments_relationships.py +33 -0
  299. wbfdm/viewsets/configs/titles/__init__.py +42 -0
  300. wbfdm/viewsets/configs/titles/classifications.py +79 -0
  301. wbfdm/viewsets/configs/titles/esg.py +11 -0
  302. wbfdm/viewsets/configs/titles/exchanges.py +12 -0
  303. wbfdm/viewsets/configs/titles/financial_ratio_analysis.py +6 -0
  304. wbfdm/viewsets/configs/titles/financials_analysis.py +50 -0
  305. wbfdm/viewsets/configs/titles/instrument_prices.py +50 -0
  306. wbfdm/viewsets/configs/titles/instrument_requests.py +16 -0
  307. wbfdm/viewsets/configs/titles/instruments.py +31 -0
  308. wbfdm/viewsets/configs/titles/instruments_relationships.py +21 -0
  309. wbfdm/viewsets/configs/titles/market_data.py +13 -0
  310. wbfdm/viewsets/configs/titles/prices.py +15 -0
  311. wbfdm/viewsets/configs/titles/statement_with_estimates.py +10 -0
  312. wbfdm/viewsets/esg.py +72 -0
  313. wbfdm/viewsets/exchanges.py +63 -0
  314. wbfdm/viewsets/financial_analysis/__init__.py +3 -0
  315. wbfdm/viewsets/financial_analysis/financial_metric_analysis.py +85 -0
  316. wbfdm/viewsets/financial_analysis/financial_ratio_analysis.py +85 -0
  317. wbfdm/viewsets/financial_analysis/statement_with_estimates.py +145 -0
  318. wbfdm/viewsets/instruments/__init__.py +80 -0
  319. wbfdm/viewsets/instruments/classifications.py +279 -0
  320. wbfdm/viewsets/instruments/financials_analysis.py +614 -0
  321. wbfdm/viewsets/instruments/instrument_lists.py +77 -0
  322. wbfdm/viewsets/instruments/instrument_prices.py +542 -0
  323. wbfdm/viewsets/instruments/instrument_requests.py +51 -0
  324. wbfdm/viewsets/instruments/instruments.py +106 -0
  325. wbfdm/viewsets/instruments/instruments_relationships.py +235 -0
  326. wbfdm/viewsets/instruments/utils.py +27 -0
  327. wbfdm/viewsets/market_data.py +172 -0
  328. wbfdm/viewsets/mixins.py +9 -0
  329. wbfdm/viewsets/officers.py +27 -0
  330. wbfdm/viewsets/prices.py +62 -0
  331. wbfdm/viewsets/statements/__init__.py +1 -0
  332. wbfdm/viewsets/statements/statements.py +100 -0
  333. wbfdm/viewsets/technical_analysis/__init__.py +1 -0
  334. wbfdm/viewsets/technical_analysis/monthly_performances.py +93 -0
  335. wbfdm-2.2.1.dist-info/METADATA +15 -0
  336. wbfdm-2.2.1.dist-info/RECORD +337 -0
  337. wbfdm-2.2.1.dist-info/WHEEL +5 -0
@@ -0,0 +1,176 @@
1
+ import magic
2
+ from django.apps import apps
3
+ from django.db.models import Q
4
+ from import_export import fields
5
+ from import_export.formats.base_formats import CSV
6
+ from import_export.results import RowResult
7
+ from import_export.signals import post_import
8
+ from import_export.widgets import CharWidget, ForeignKeyWidget, ManyToManyWidget
9
+ from rest_framework.reverse import reverse
10
+ from wbcore.contrib.currency.models import Currency
11
+ from wbcore.contrib.geography.import_export.resources.geography import (
12
+ CountryForeignKeyWidget,
13
+ )
14
+ from wbcore.contrib.io.resources import FilterModelResource
15
+ from wbcore.contrib.notifications.dispatch import send_notification
16
+ from wbfdm.models import Exchange, Instrument, InstrumentClassificationThroughModel
17
+
18
+ from .classification import ClassificationManyToManyWidget
19
+
20
+
21
+ class TypeWidget(CharWidget):
22
+ def clean(self, value, row=None, **kwargs):
23
+ return super().clean(value.lower().name(), row=row, **kwargs)
24
+
25
+
26
+ class InstrumentResource(FilterModelResource):
27
+ """
28
+ The resource to download AssetPositions
29
+ """
30
+
31
+ instrument_type = fields.Field(column_name="instrument_type", attribute="instrument_type", widget=TypeWidget())
32
+
33
+ country = fields.Field(
34
+ column_name="country",
35
+ attribute="country",
36
+ widget=CountryForeignKeyWidget(),
37
+ )
38
+
39
+ exchanges = fields.Field(
40
+ column_name="exchanges",
41
+ attribute="exchanges",
42
+ m2m_add=True,
43
+ widget=ManyToManyWidget(Exchange, field="mic_code"),
44
+ )
45
+
46
+ currency = fields.Field(
47
+ column_name="currency",
48
+ attribute="currency",
49
+ widget=ForeignKeyWidget(Currency, field="key__iexact"),
50
+ )
51
+
52
+ primary_classifications = fields.Field(
53
+ column_name="primary_classifications",
54
+ attribute="classifications",
55
+ m2m_add=True,
56
+ widget=ClassificationManyToManyWidget(primary_classification_group=True),
57
+ )
58
+ default_classifications = fields.Field(
59
+ column_name="default_classifications",
60
+ attribute="classifications",
61
+ m2m_add=True,
62
+ widget=ClassificationManyToManyWidget(primary_classification_group=False),
63
+ )
64
+
65
+ def save_m2m(self, obj, data, using_transactions, dry_run):
66
+ """
67
+ override save m2m to define what to to with the nested classification instrument through model values
68
+ """
69
+ if (not using_transactions and dry_run) or self._meta.use_bulk:
70
+ # we don't have transactions and we want to do a dry_run
71
+ # OR use_bulk is enabled (m2m operations are not supported for bulk operations)
72
+ pass
73
+ else:
74
+ for field in self.get_import_fields():
75
+ if isinstance(field.widget, ClassificationManyToManyWidget):
76
+ defaults = {
77
+ "is_favorite": str(data.get(f"{field.column_name}__is_favorite", "")).lower()
78
+ in ["1", "true", "yes"],
79
+ "reason": data.get(f"{field.column_name}__reason", None),
80
+ "pure_player": data.get(f"{field.column_name}__pure_player", None),
81
+ "top_player": data.get(f"{field.column_name}__top_player", None),
82
+ "percent_of_revenue": data.get(f"{field.column_name}__percent_of_revenue", None),
83
+ }
84
+ defaults = {k: v for k, v in defaults.items() if v is not None}
85
+ for classification in field.clean(data):
86
+ InstrumentClassificationThroughModel.objects.update_or_create(
87
+ instrument=obj, classification=classification, defaults=defaults
88
+ )
89
+ elif isinstance(field.widget, ManyToManyWidget):
90
+ self.import_field(field, obj, data, True)
91
+
92
+ def get_instance(self, instance_loader, row):
93
+ try:
94
+ return Instrument.objects.get(
95
+ Q(isin=row["isin"]) | Q(refinitiv_identifier_code=row["refinitiv_identifier_code"])
96
+ )
97
+ except Instrument.DoesNotExist:
98
+ return None
99
+
100
+ class Meta:
101
+ import_id_fields = ("isin",)
102
+ fields = (
103
+ "id",
104
+ "founded_year",
105
+ "inception_date",
106
+ "delisted_date",
107
+ "identifier",
108
+ "name",
109
+ "name_repr",
110
+ "description",
111
+ "isin",
112
+ "ticker",
113
+ "refinitiv_ticker",
114
+ "refinitiv_identifier_code",
115
+ "refinitiv_mnemonic_code",
116
+ "sedol",
117
+ "valoren",
118
+ "headquarter_address",
119
+ "primary_url",
120
+ "is_cash",
121
+ "instrument_type",
122
+ "country",
123
+ "exchanges",
124
+ "currency",
125
+ "primary_classifications",
126
+ "default_classifications",
127
+ )
128
+ export_order = fields
129
+ model = Instrument
130
+
131
+
132
+ # This parser satisfy the import source framework interface. The generic logic will probably need to move the io module to avoid repetition
133
+ def parse(import_source):
134
+ input_format = CSV(encoding="utf-8-sig")
135
+
136
+ file_stream = import_source.file.read()
137
+ if input_format.CONTENT_TYPE == magic.from_buffer(file_stream, mime=True):
138
+ dataset = input_format.create_dataset(file_stream)
139
+
140
+ model = apps.get_model(import_source.parser_handler.handler)
141
+ result = InstrumentResource().import_data(
142
+ dataset,
143
+ dry_run=False,
144
+ file_name=import_source.file.name,
145
+ user=import_source.creator,
146
+ rollback_on_validation_errors=True,
147
+ )
148
+ import_source.file.close()
149
+ post_import.send(sender=None, model=model)
150
+ success_message = """
151
+ {} import finished:
152
+ * new {}
153
+ * updated {}
154
+ * skipped {}
155
+ * failed {}
156
+ * deleted {}
157
+ * invalid {}
158
+ """.format(
159
+ model._meta.verbose_name_plural,
160
+ result.totals[RowResult.IMPORT_TYPE_NEW],
161
+ result.totals[RowResult.IMPORT_TYPE_UPDATE],
162
+ result.totals[RowResult.IMPORT_TYPE_SKIP],
163
+ result.totals[RowResult.IMPORT_TYPE_ERROR],
164
+ result.totals[RowResult.IMPORT_TYPE_DELETE],
165
+ result.totals[RowResult.IMPORT_TYPE_INVALID],
166
+ )
167
+ import_source.log = success_message
168
+ import_source.save()
169
+ if import_source.creator:
170
+ send_notification(
171
+ code="io.import_done",
172
+ name="Your import is done",
173
+ body=success_message,
174
+ user=import_source.creator,
175
+ endpoint=reverse(f"{model.get_base_endpoint()}-list") if hasattr(model, "get_base_endpoint") else None,
176
+ )
wbfdm/jinja2.py ADDED
@@ -0,0 +1,7 @@
1
+ from jinja2 import Environment
2
+ from jinjasql import JinjaSql # type: ignore
3
+
4
+
5
+ def get_environment(**options):
6
+ env = Environment(**options)
7
+ return JinjaSql(env=env, param_style="format").env
@@ -0,0 +1,30 @@
1
+ from django_celery_beat.models import IntervalSchedule, PeriodicTask
2
+ from django.db import DEFAULT_DB_ALIAS
3
+ from django.apps import apps as global_apps
4
+
5
+
6
+ def initialize_task(app_config, verbosity=2, interactive=True, using=DEFAULT_DB_ALIAS, apps=global_apps, **kwargs):
7
+ PeriodicTask.objects.get_or_create(
8
+ task="wbfdm.tasks.update_of_investable_universe_data",
9
+ defaults={
10
+ "name": "FDM: Periodic data update of investable universe",
11
+ "interval": IntervalSchedule.objects.get_or_create(every=3, period=IntervalSchedule.HOURS)[0],
12
+ "crontab": None,
13
+ },
14
+ )
15
+ PeriodicTask.objects.get_or_create(
16
+ task="wbfdm.tasks.synchronize_instruments_as_task",
17
+ defaults={
18
+ "name": "FDM: Synchronize Instrument",
19
+ "interval": IntervalSchedule.objects.get_or_create(every=30, period=IntervalSchedule.MINUTES)[0],
20
+ "crontab": None,
21
+ },
22
+ )
23
+ PeriodicTask.objects.get_or_create(
24
+ task="wbfdm.tasks.synchronize_exchanges_as_task",
25
+ defaults={
26
+ "name": "FDM: Synchronize Exchange",
27
+ "interval": IntervalSchedule.objects.get_or_create(every=30, period=IntervalSchedule.MINUTES)[0],
28
+ "crontab": None,
29
+ },
30
+ )
wbfdm/menu.py ADDED
@@ -0,0 +1,11 @@
1
+ from django.utils.translation import gettext as _
2
+ from wbcore.menus import ItemPermission, MenuItem
3
+ from wbcore.permissions.shortcuts import is_internal_user
4
+
5
+ INSTRUMENT_MENU_ITEM = MenuItem(
6
+ label=_("Instruments"),
7
+ endpoint="wbfdm:instrument-list",
8
+ permission=ItemPermission(
9
+ method=lambda request: is_internal_user(request.user), permissions=["wbfdm.view_instrument"]
10
+ ),
11
+ )
@@ -0,0 +1,71 @@
1
+ # Generated by Django 4.1.9 on 2023-06-14 07:31
2
+
3
+ import django.db.models.deletion
4
+ from django.db import migrations, models
5
+
6
+
7
+ class Migration(migrations.Migration):
8
+ initial = True
9
+
10
+ dependencies = []
11
+
12
+ operations = [
13
+ migrations.CreateModel(
14
+ name="Instrument",
15
+ fields=[
16
+ ("id", models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")),
17
+ ("name", models.CharField(max_length=512)),
18
+ ],
19
+ options={
20
+ "verbose_name": "Instrument",
21
+ "verbose_name_plural": "Instruments",
22
+ },
23
+ ),
24
+ migrations.CreateModel(
25
+ name="InstrumentLookup",
26
+ fields=[
27
+ ("id", models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")),
28
+ ("identifier", models.CharField(max_length=64)),
29
+ ("backend_path", models.CharField(max_length=265)),
30
+ ("prices", models.BooleanField(default=False)),
31
+ ("dividends", models.BooleanField(default=False)),
32
+ ("corporate_actions", models.BooleanField(default=False)),
33
+ ("statements", models.BooleanField(default=False)),
34
+ ("estimates", models.BooleanField(default=False)),
35
+ ("officers", models.BooleanField(default=False)),
36
+ (
37
+ "instrument",
38
+ models.ForeignKey(
39
+ on_delete=django.db.models.deletion.CASCADE, related_name="lookups", to="wbfdm.instrument"
40
+ ),
41
+ ),
42
+ ],
43
+ ),
44
+ migrations.CreateModel(
45
+ name="InstrumentIdentifier",
46
+ fields=[
47
+ ("id", models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name="ID")),
48
+ ("identifier", models.CharField(max_length=32)),
49
+ (
50
+ "identifier_type",
51
+ models.CharField(
52
+ choices=[
53
+ ("ISIN", "ISIN"),
54
+ ("RIC", "RIC"),
55
+ ("BBG", "BBG"),
56
+ ("TICKER", "Ticker"),
57
+ ("VALOREN", "Valoren"),
58
+ ("WKN", "WKN"),
59
+ ],
60
+ max_length=32,
61
+ ),
62
+ ),
63
+ (
64
+ "instrument",
65
+ models.ForeignKey(
66
+ on_delete=django.db.models.deletion.CASCADE, related_name="identifiers", to="wbfdm.instrument"
67
+ ),
68
+ ),
69
+ ],
70
+ ),
71
+ ]
@@ -0,0 +1,144 @@
1
+ # Generated by Django 4.1.9 on 2023-06-23 11:51
2
+
3
+ import django.db.models.deletion
4
+ import wbfdm.models.fk_fields
5
+ from django.db import migrations, models
6
+
7
+
8
+ class Migration(migrations.Migration):
9
+ dependencies = [
10
+ ("wbfdm", "0001_initial"),
11
+ ]
12
+
13
+ operations = [
14
+ migrations.RenameField(
15
+ model_name="instrumentlookup",
16
+ old_name="statements",
17
+ new_name="financials",
18
+ ),
19
+ migrations.AddField(
20
+ model_name="instrument",
21
+ name="bbg_lookup",
22
+ field=wbfdm.models.fk_fields.CompositeForeignKey(
23
+ null=True,
24
+ null_if_equal=[],
25
+ on_delete=django.db.models.deletion.SET_NULL,
26
+ related_name="+",
27
+ to="wbfdm.instrumentidentifier",
28
+ to_fields={
29
+ "identifier_type": wbfdm.models.fk_fields.RawFieldValue("BBG"),
30
+ "instrument": wbfdm.models.fk_fields.LocalFieldValue("id"),
31
+ },
32
+ ),
33
+ ),
34
+ migrations.AddField(
35
+ model_name="instrument",
36
+ name="financial_backend",
37
+ field=wbfdm.models.fk_fields.CompositeForeignKey(
38
+ null=True,
39
+ null_if_equal=[],
40
+ on_delete=django.db.models.deletion.DO_NOTHING,
41
+ related_name="+",
42
+ to="wbfdm.instrumentlookup",
43
+ to_fields={
44
+ "financials": wbfdm.models.fk_fields.RawFieldValue(True),
45
+ "instrument_id": wbfdm.models.fk_fields.LocalFieldValue("id"),
46
+ },
47
+ ),
48
+ ),
49
+ migrations.AddField(
50
+ model_name="instrument",
51
+ name="isin_lookup",
52
+ field=wbfdm.models.fk_fields.CompositeForeignKey(
53
+ null=True,
54
+ null_if_equal=[],
55
+ on_delete=django.db.models.deletion.SET_NULL,
56
+ related_name="+",
57
+ to="wbfdm.instrumentidentifier",
58
+ to_fields={
59
+ "identifier_type": wbfdm.models.fk_fields.RawFieldValue("ISIN"),
60
+ "instrument": wbfdm.models.fk_fields.LocalFieldValue("id"),
61
+ },
62
+ ),
63
+ ),
64
+ migrations.AddField(
65
+ model_name="instrument",
66
+ name="price_backend",
67
+ field=wbfdm.models.fk_fields.CompositeForeignKey(
68
+ null=True,
69
+ null_if_equal=[],
70
+ on_delete=django.db.models.deletion.DO_NOTHING,
71
+ related_name="+",
72
+ to="wbfdm.instrumentlookup",
73
+ to_fields={
74
+ "instrument_id": wbfdm.models.fk_fields.LocalFieldValue("id"),
75
+ "prices": wbfdm.models.fk_fields.RawFieldValue(True),
76
+ },
77
+ ),
78
+ ),
79
+ migrations.AddField(
80
+ model_name="instrument",
81
+ name="ric_lookup",
82
+ field=wbfdm.models.fk_fields.CompositeForeignKey(
83
+ null=True,
84
+ null_if_equal=[],
85
+ on_delete=django.db.models.deletion.SET_NULL,
86
+ related_name="+",
87
+ to="wbfdm.instrumentidentifier",
88
+ to_fields={
89
+ "identifier_type": wbfdm.models.fk_fields.RawFieldValue("RIC"),
90
+ "instrument": wbfdm.models.fk_fields.LocalFieldValue("id"),
91
+ },
92
+ ),
93
+ ),
94
+ migrations.AddField(
95
+ model_name="instrument",
96
+ name="ticker_lookup",
97
+ field=wbfdm.models.fk_fields.CompositeForeignKey(
98
+ null=True,
99
+ null_if_equal=[],
100
+ on_delete=django.db.models.deletion.SET_NULL,
101
+ related_name="+",
102
+ to="wbfdm.instrumentidentifier",
103
+ to_fields={
104
+ "identifier_type": wbfdm.models.fk_fields.RawFieldValue("TICKER"),
105
+ "instrument": wbfdm.models.fk_fields.LocalFieldValue("id"),
106
+ },
107
+ ),
108
+ ),
109
+ migrations.AddField(
110
+ model_name="instrument",
111
+ name="valoren_lookup",
112
+ field=wbfdm.models.fk_fields.CompositeForeignKey(
113
+ null=True,
114
+ null_if_equal=[],
115
+ on_delete=django.db.models.deletion.SET_NULL,
116
+ related_name="+",
117
+ to="wbfdm.instrumentidentifier",
118
+ to_fields={
119
+ "identifier_type": wbfdm.models.fk_fields.RawFieldValue("VALOREN"),
120
+ "instrument": wbfdm.models.fk_fields.LocalFieldValue("id"),
121
+ },
122
+ ),
123
+ ),
124
+ migrations.AddField(
125
+ model_name="instrument",
126
+ name="wkn_lookup",
127
+ field=wbfdm.models.fk_fields.CompositeForeignKey(
128
+ null=True,
129
+ null_if_equal=[],
130
+ on_delete=django.db.models.deletion.SET_NULL,
131
+ related_name="+",
132
+ to="wbfdm.instrumentidentifier",
133
+ to_fields={
134
+ "identifier_type": wbfdm.models.fk_fields.RawFieldValue("WKN"),
135
+ "instrument": wbfdm.models.fk_fields.LocalFieldValue("id"),
136
+ },
137
+ ),
138
+ ),
139
+ migrations.AddField(
140
+ model_name="instrumentlookup",
141
+ name="parameters",
142
+ field=models.JSONField(default=dict, null=True),
143
+ ),
144
+ ]
@@ -0,0 +1,34 @@
1
+ # Generated by Django 4.1.9 on 2023-07-11 08:24
2
+
3
+ import django.db.models.deletion
4
+ import wbfdm.models.fk_fields
5
+ from django.db import migrations, models
6
+
7
+
8
+ class Migration(migrations.Migration):
9
+ dependencies = [
10
+ ("wbfdm", "0002_rename_statements_instrumentlookup_financials_and_more"),
11
+ ]
12
+
13
+ operations = [
14
+ migrations.AddField(
15
+ model_name="instrument",
16
+ name="estimate_backend",
17
+ field=wbfdm.models.fk_fields.CompositeForeignKey(
18
+ null=True,
19
+ null_if_equal=[],
20
+ on_delete=django.db.models.deletion.DO_NOTHING,
21
+ related_name="+",
22
+ to="wbfdm.instrumentlookup",
23
+ to_fields={
24
+ "estimates": wbfdm.models.fk_fields.RawFieldValue(True),
25
+ "instrument_id": wbfdm.models.fk_fields.LocalFieldValue("id"),
26
+ },
27
+ ),
28
+ ),
29
+ migrations.AlterField(
30
+ model_name="instrumentlookup",
31
+ name="parameters",
32
+ field=models.JSONField(blank=True, default=dict, null=True),
33
+ ),
34
+ ]
@@ -0,0 +1,86 @@
1
+ # Generated by Django 4.1.9 on 2023-07-12 14:44
2
+
3
+ import django.db.models.deletion
4
+ import wbfdm.models.fk_fields
5
+ from django.db import migrations
6
+
7
+
8
+ class Migration(migrations.Migration):
9
+ dependencies = [
10
+ ("wbfdm", "0003_instrument_estimate_backend_and_more"),
11
+ ]
12
+
13
+ operations = [
14
+ migrations.RenameField(
15
+ model_name="instrumentlookup",
16
+ old_name="financials",
17
+ new_name="statements",
18
+ ),
19
+ migrations.RemoveField(
20
+ model_name="instrument",
21
+ name="financial_backend",
22
+ ),
23
+ migrations.AddField(
24
+ model_name="instrument",
25
+ name="statement_backend",
26
+ field=wbfdm.models.fk_fields.CompositeForeignKey(
27
+ null=True,
28
+ null_if_equal=[],
29
+ on_delete=django.db.models.deletion.DO_NOTHING,
30
+ related_name="+",
31
+ to="wbfdm.instrumentlookup",
32
+ to_fields={
33
+ "instrument_id": wbfdm.models.fk_fields.LocalFieldValue("id"),
34
+ "statements": wbfdm.models.fk_fields.RawFieldValue(True),
35
+ },
36
+ ),
37
+ ),
38
+ migrations.RenameField(
39
+ model_name="instrumentlookup",
40
+ old_name="estimates",
41
+ new_name="financials",
42
+ ),
43
+ migrations.RemoveField(
44
+ model_name="instrument",
45
+ name="estimate_backend",
46
+ ),
47
+ migrations.AddField(
48
+ model_name="instrument",
49
+ name="financial_backend",
50
+ field=wbfdm.models.fk_fields.CompositeForeignKey(
51
+ null=True,
52
+ null_if_equal=[],
53
+ on_delete=django.db.models.deletion.DO_NOTHING,
54
+ related_name="+",
55
+ to="wbfdm.instrumentlookup",
56
+ to_fields={
57
+ "financials": wbfdm.models.fk_fields.RawFieldValue(True),
58
+ "instrument_id": wbfdm.models.fk_fields.LocalFieldValue("id"),
59
+ },
60
+ ),
61
+ ),
62
+ migrations.RenameField(
63
+ model_name="instrumentlookup",
64
+ old_name="prices",
65
+ new_name="market_data",
66
+ ),
67
+ migrations.RemoveField(
68
+ model_name="instrument",
69
+ name="price_backend",
70
+ ),
71
+ migrations.AddField(
72
+ model_name="instrument",
73
+ name="market_data_backend",
74
+ field=wbfdm.models.fk_fields.CompositeForeignKey(
75
+ null=True,
76
+ null_if_equal=[],
77
+ on_delete=django.db.models.deletion.DO_NOTHING,
78
+ related_name="+",
79
+ to="wbfdm.instrumentlookup",
80
+ to_fields={
81
+ "instrument_id": wbfdm.models.fk_fields.LocalFieldValue("id"),
82
+ "market_data": wbfdm.models.fk_fields.RawFieldValue(True),
83
+ },
84
+ ),
85
+ ),
86
+ ]
@@ -0,0 +1,29 @@
1
+ # Generated by Django 4.1.9 on 2023-07-19 07:42
2
+
3
+ import django.db.models.deletion
4
+ import wbfdm.models.fk_fields
5
+ from django.db import migrations
6
+
7
+
8
+ class Migration(migrations.Migration):
9
+ dependencies = [
10
+ ("wbfdm", "0004_rename_financials_instrumentlookup_statements_and_more"),
11
+ ]
12
+
13
+ operations = [
14
+ migrations.AddField(
15
+ model_name="instrument",
16
+ name="corporate_action_backend",
17
+ field=wbfdm.models.fk_fields.CompositeForeignKey(
18
+ null=True,
19
+ null_if_equal=[],
20
+ on_delete=django.db.models.deletion.DO_NOTHING,
21
+ related_name="+",
22
+ to="wbfdm.instrumentlookup",
23
+ to_fields={
24
+ "corporate_actions": wbfdm.models.fk_fields.RawFieldValue(True),
25
+ "instrument_id": wbfdm.models.fk_fields.LocalFieldValue("id"),
26
+ },
27
+ ),
28
+ ),
29
+ ]
@@ -0,0 +1,29 @@
1
+ # Generated by Django 4.1.11 on 2023-10-04 06:48
2
+
3
+ import django.db.models.deletion
4
+ import wbfdm.models.fk_fields
5
+ from django.db import migrations
6
+
7
+
8
+ class Migration(migrations.Migration):
9
+ dependencies = [
10
+ ("wbfdm", "0005_instrument_corporate_action_backend"),
11
+ ]
12
+
13
+ operations = [
14
+ migrations.AddField(
15
+ model_name="instrument",
16
+ name="officer_backend",
17
+ field=wbfdm.models.fk_fields.CompositeForeignKey(
18
+ null=True,
19
+ null_if_equal=[],
20
+ on_delete=django.db.models.deletion.DO_NOTHING,
21
+ related_name="+",
22
+ to="wbfdm.instrumentlookup",
23
+ to_fields={
24
+ "instrument_id": wbfdm.models.fk_fields.LocalFieldValue("id"),
25
+ "officers": wbfdm.models.fk_fields.RawFieldValue(True),
26
+ },
27
+ ),
28
+ ),
29
+ ]