wbportfolio 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 wbportfolio might be problematic. Click here for more details.

Files changed (486) hide show
  1. wbportfolio/__init__.py +1 -0
  2. wbportfolio/admin/__init__.py +12 -0
  3. wbportfolio/admin/asset.py +47 -0
  4. wbportfolio/admin/custodians.py +9 -0
  5. wbportfolio/admin/portfolio.py +127 -0
  6. wbportfolio/admin/portfolio_relationships.py +22 -0
  7. wbportfolio/admin/product_groups.py +42 -0
  8. wbportfolio/admin/products.py +80 -0
  9. wbportfolio/admin/reconciliations.py +14 -0
  10. wbportfolio/admin/registers.py +17 -0
  11. wbportfolio/admin/roles.py +19 -0
  12. wbportfolio/admin/synchronization/__init__.py +2 -0
  13. wbportfolio/admin/synchronization/admin.py +114 -0
  14. wbportfolio/admin/synchronization/portfolio_synchronization.py +18 -0
  15. wbportfolio/admin/synchronization/price_computation.py +21 -0
  16. wbportfolio/admin/transactions/__init__.py +5 -0
  17. wbportfolio/admin/transactions/claim.py +16 -0
  18. wbportfolio/admin/transactions/dividends.py +14 -0
  19. wbportfolio/admin/transactions/fees.py +35 -0
  20. wbportfolio/admin/transactions/trades.py +49 -0
  21. wbportfolio/admin/transactions/transactions.py +37 -0
  22. wbportfolio/analysis/__init__.py +0 -0
  23. wbportfolio/analysis/claims.py +235 -0
  24. wbportfolio/apps.py +5 -0
  25. wbportfolio/contrib/__init__.py +0 -0
  26. wbportfolio/contrib/company_portfolio/__init__.py +0 -0
  27. wbportfolio/contrib/company_portfolio/admin.py +28 -0
  28. wbportfolio/contrib/company_portfolio/apps.py +29 -0
  29. wbportfolio/contrib/company_portfolio/configs/__init__.py +3 -0
  30. wbportfolio/contrib/company_portfolio/configs/display.py +182 -0
  31. wbportfolio/contrib/company_portfolio/configs/endpoints.py +34 -0
  32. wbportfolio/contrib/company_portfolio/configs/previews.py +37 -0
  33. wbportfolio/contrib/company_portfolio/constants.py +1 -0
  34. wbportfolio/contrib/company_portfolio/dynamic_preferences_registry.py +87 -0
  35. wbportfolio/contrib/company_portfolio/factories.py +32 -0
  36. wbportfolio/contrib/company_portfolio/filters.py +127 -0
  37. wbportfolio/contrib/company_portfolio/management.py +19 -0
  38. wbportfolio/contrib/company_portfolio/migrations/0001_initial.py +214 -0
  39. wbportfolio/contrib/company_portfolio/migrations/__init__.py +0 -0
  40. wbportfolio/contrib/company_portfolio/models.py +334 -0
  41. wbportfolio/contrib/company_portfolio/scripts.py +76 -0
  42. wbportfolio/contrib/company_portfolio/serializers.py +303 -0
  43. wbportfolio/contrib/company_portfolio/tasks.py +19 -0
  44. wbportfolio/contrib/company_portfolio/tests/__init__.py +0 -0
  45. wbportfolio/contrib/company_portfolio/tests/conftest.py +161 -0
  46. wbportfolio/contrib/company_portfolio/tests/test_models.py +161 -0
  47. wbportfolio/contrib/company_portfolio/urls.py +29 -0
  48. wbportfolio/contrib/company_portfolio/viewsets.py +195 -0
  49. wbportfolio/defaults/__init__.py +0 -0
  50. wbportfolio/defaults/fees/__init__.py +0 -0
  51. wbportfolio/defaults/fees/default.py +92 -0
  52. wbportfolio/defaults/portfolio/__init__.py +0 -0
  53. wbportfolio/defaults/portfolio/default_rebalancing.py +45 -0
  54. wbportfolio/dynamic_preferences_registry.py +58 -0
  55. wbportfolio/factories/__init__.py +35 -0
  56. wbportfolio/factories/adjustments.py +17 -0
  57. wbportfolio/factories/assets.py +75 -0
  58. wbportfolio/factories/claim.py +39 -0
  59. wbportfolio/factories/custodians.py +11 -0
  60. wbportfolio/factories/dividends.py +14 -0
  61. wbportfolio/factories/fees.py +15 -0
  62. wbportfolio/factories/indexes.py +17 -0
  63. wbportfolio/factories/portfolio_cash_flow.py +20 -0
  64. wbportfolio/factories/portfolio_cash_targets.py +15 -0
  65. wbportfolio/factories/portfolio_swing_pricings.py +15 -0
  66. wbportfolio/factories/portfolios.py +59 -0
  67. wbportfolio/factories/product_groups.py +28 -0
  68. wbportfolio/factories/products.py +56 -0
  69. wbportfolio/factories/pytest_utils.py +121 -0
  70. wbportfolio/factories/reconciliations.py +23 -0
  71. wbportfolio/factories/roles.py +20 -0
  72. wbportfolio/factories/synchronization.py +40 -0
  73. wbportfolio/factories/trades.py +35 -0
  74. wbportfolio/factories/transactions.py +21 -0
  75. wbportfolio/fdm/__init__.py +0 -0
  76. wbportfolio/fdm/tasks.py +12 -0
  77. wbportfolio/filters/__init__.py +32 -0
  78. wbportfolio/filters/assets.py +485 -0
  79. wbportfolio/filters/assets_and_net_new_money_progression.py +42 -0
  80. wbportfolio/filters/custodians.py +10 -0
  81. wbportfolio/filters/esg.py +22 -0
  82. wbportfolio/filters/performances.py +171 -0
  83. wbportfolio/filters/portfolios.py +24 -0
  84. wbportfolio/filters/positions.py +178 -0
  85. wbportfolio/filters/products.py +157 -0
  86. wbportfolio/filters/roles.py +26 -0
  87. wbportfolio/filters/signals.py +92 -0
  88. wbportfolio/filters/transactions/__init__.py +20 -0
  89. wbportfolio/filters/transactions/claim.py +394 -0
  90. wbportfolio/filters/transactions/fees.py +66 -0
  91. wbportfolio/filters/transactions/trades.py +224 -0
  92. wbportfolio/filters/transactions/transactions.py +98 -0
  93. wbportfolio/import_export/__init__.py +0 -0
  94. wbportfolio/import_export/backends/__init__.py +2 -0
  95. wbportfolio/import_export/backends/ubs/__init__.py +3 -0
  96. wbportfolio/import_export/backends/ubs/asset_position.py +45 -0
  97. wbportfolio/import_export/backends/ubs/fees.py +63 -0
  98. wbportfolio/import_export/backends/ubs/instrument_price.py +44 -0
  99. wbportfolio/import_export/backends/ubs/mixin.py +15 -0
  100. wbportfolio/import_export/backends/utils.py +58 -0
  101. wbportfolio/import_export/backends/wbfdm/__init__.py +2 -0
  102. wbportfolio/import_export/backends/wbfdm/adjustment.py +50 -0
  103. wbportfolio/import_export/backends/wbfdm/dividend.py +16 -0
  104. wbportfolio/import_export/backends/wbfdm/mixin.py +15 -0
  105. wbportfolio/import_export/handlers/__init__.py +0 -0
  106. wbportfolio/import_export/handlers/adjustment.py +39 -0
  107. wbportfolio/import_export/handlers/asset_position.py +167 -0
  108. wbportfolio/import_export/handlers/dividend.py +80 -0
  109. wbportfolio/import_export/handlers/fees.py +58 -0
  110. wbportfolio/import_export/handlers/portfolio_cash_flow.py +57 -0
  111. wbportfolio/import_export/handlers/register.py +43 -0
  112. wbportfolio/import_export/handlers/trade.py +191 -0
  113. wbportfolio/import_export/parsers/__init__.py +0 -0
  114. wbportfolio/import_export/parsers/default_mapping.py +30 -0
  115. wbportfolio/import_export/parsers/jpmorgan/__init__.py +0 -0
  116. wbportfolio/import_export/parsers/jpmorgan/customer_trade.py +63 -0
  117. wbportfolio/import_export/parsers/jpmorgan/fees.py +64 -0
  118. wbportfolio/import_export/parsers/jpmorgan/strategy.py +116 -0
  119. wbportfolio/import_export/parsers/jpmorgan/valuation.py +41 -0
  120. wbportfolio/import_export/parsers/leonteq/__init__.py +0 -0
  121. wbportfolio/import_export/parsers/leonteq/customer_trade.py +47 -0
  122. wbportfolio/import_export/parsers/leonteq/equity.py +81 -0
  123. wbportfolio/import_export/parsers/leonteq/fees.py +70 -0
  124. wbportfolio/import_export/parsers/leonteq/trade.py +94 -0
  125. wbportfolio/import_export/parsers/leonteq/valuation.py +39 -0
  126. wbportfolio/import_export/parsers/natixis/__init__.py +0 -0
  127. wbportfolio/import_export/parsers/natixis/customer_trade.py +62 -0
  128. wbportfolio/import_export/parsers/natixis/d1_customer_trade.py +66 -0
  129. wbportfolio/import_export/parsers/natixis/d1_equity.py +80 -0
  130. wbportfolio/import_export/parsers/natixis/d1_fees.py +58 -0
  131. wbportfolio/import_export/parsers/natixis/d1_trade.py +70 -0
  132. wbportfolio/import_export/parsers/natixis/d1_valuation.py +41 -0
  133. wbportfolio/import_export/parsers/natixis/dividend.py +53 -0
  134. wbportfolio/import_export/parsers/natixis/equity.py +60 -0
  135. wbportfolio/import_export/parsers/natixis/fees.py +53 -0
  136. wbportfolio/import_export/parsers/natixis/trade.py +63 -0
  137. wbportfolio/import_export/parsers/natixis/utils.py +76 -0
  138. wbportfolio/import_export/parsers/natixis/valuation.py +46 -0
  139. wbportfolio/import_export/parsers/refinitiv/__init__.py +0 -0
  140. wbportfolio/import_export/parsers/refinitiv/adjustment.py +24 -0
  141. wbportfolio/import_export/parsers/sg_lux/__init__.py +0 -0
  142. wbportfolio/import_export/parsers/sg_lux/custodian_positions.py +70 -0
  143. wbportfolio/import_export/parsers/sg_lux/customer_trade.py +75 -0
  144. wbportfolio/import_export/parsers/sg_lux/customer_trade_pending_slk.py +140 -0
  145. wbportfolio/import_export/parsers/sg_lux/customer_trade_slk.py +80 -0
  146. wbportfolio/import_export/parsers/sg_lux/customer_trade_without_pw.py +57 -0
  147. wbportfolio/import_export/parsers/sg_lux/equity.py +137 -0
  148. wbportfolio/import_export/parsers/sg_lux/fees.py +56 -0
  149. wbportfolio/import_export/parsers/sg_lux/perf_fees.py +51 -0
  150. wbportfolio/import_export/parsers/sg_lux/portfolio_cash_flow.py +29 -0
  151. wbportfolio/import_export/parsers/sg_lux/portfolio_future_cash_flow.py +36 -0
  152. wbportfolio/import_export/parsers/sg_lux/registers.py +210 -0
  153. wbportfolio/import_export/parsers/sg_lux/sylk.py +248 -0
  154. wbportfolio/import_export/parsers/sg_lux/utils.py +36 -0
  155. wbportfolio/import_export/parsers/sg_lux/valuation.py +53 -0
  156. wbportfolio/import_export/parsers/societe_generale/__init__.py +0 -0
  157. wbportfolio/import_export/parsers/societe_generale/customer_trade.py +54 -0
  158. wbportfolio/import_export/parsers/societe_generale/strategy.py +94 -0
  159. wbportfolio/import_export/parsers/societe_generale/valuation.py +37 -0
  160. wbportfolio/import_export/parsers/tellco/__init__.py +0 -0
  161. wbportfolio/import_export/parsers/tellco/customer_trade.py +64 -0
  162. wbportfolio/import_export/parsers/tellco/equity.py +86 -0
  163. wbportfolio/import_export/parsers/tellco/valuation.py +52 -0
  164. wbportfolio/import_export/parsers/ubs/__init__.py +0 -0
  165. wbportfolio/import_export/parsers/ubs/api/__init__.py +0 -0
  166. wbportfolio/import_export/parsers/ubs/api/asset_position.py +106 -0
  167. wbportfolio/import_export/parsers/ubs/api/fees.py +31 -0
  168. wbportfolio/import_export/parsers/ubs/api/instrument_price.py +20 -0
  169. wbportfolio/import_export/parsers/ubs/api/utils.py +0 -0
  170. wbportfolio/import_export/parsers/ubs/customer_trade.py +60 -0
  171. wbportfolio/import_export/parsers/ubs/equity.py +97 -0
  172. wbportfolio/import_export/parsers/ubs/historical_customer_trade.py +67 -0
  173. wbportfolio/import_export/parsers/ubs/valuation.py +52 -0
  174. wbportfolio/import_export/parsers/vontobel/__init__.py +0 -0
  175. wbportfolio/import_export/parsers/vontobel/asset_position.py +97 -0
  176. wbportfolio/import_export/parsers/vontobel/customer_trade.py +54 -0
  177. wbportfolio/import_export/parsers/vontobel/historical_customer_trade.py +40 -0
  178. wbportfolio/import_export/parsers/vontobel/instrument.py +34 -0
  179. wbportfolio/import_export/parsers/vontobel/management_fees.py +86 -0
  180. wbportfolio/import_export/parsers/vontobel/performance_fees.py +35 -0
  181. wbportfolio/import_export/parsers/vontobel/trade.py +38 -0
  182. wbportfolio/import_export/parsers/vontobel/utils.py +17 -0
  183. wbportfolio/import_export/parsers/vontobel/valuation.py +29 -0
  184. wbportfolio/import_export/resources/__init__.py +0 -0
  185. wbportfolio/import_export/resources/assets.py +68 -0
  186. wbportfolio/import_export/resources/trades.py +41 -0
  187. wbportfolio/import_export/utils.py +42 -0
  188. wbportfolio/metric/__init__.py +0 -0
  189. wbportfolio/metric/backends/__init__.py +2 -0
  190. wbportfolio/metric/backends/base.py +86 -0
  191. wbportfolio/metric/backends/constants.py +222 -0
  192. wbportfolio/metric/backends/portfolio_base.py +255 -0
  193. wbportfolio/metric/backends/portfolio_esg.py +66 -0
  194. wbportfolio/metric/tests/__init__.py +0 -0
  195. wbportfolio/metric/tests/conftest.py +4 -0
  196. wbportfolio/metric/tests/test_portfolio_base.py +135 -0
  197. wbportfolio/metric/tests/test_portfolio_esg.py +69 -0
  198. wbportfolio/migrations/0001_initial_squashed.py +13848 -0
  199. wbportfolio/migrations/0002_product_default_sub_account_squashed_0039_alter_assetallocation_company_and_more.py +3836 -0
  200. wbportfolio/migrations/0040_instrument_financial_instrument.py +26 -0
  201. wbportfolio/migrations/0041_remove_listresearch_research_ptr_and_more.py +129 -0
  202. wbportfolio/migrations/0042_instrumentlist_instrumentlistthroughmodel_and_more.py +71 -0
  203. wbportfolio/migrations/0043_alter_instrumentlistthroughmodel_options_and_more.py +238 -0
  204. wbportfolio/migrations/0044_alter_instrumentlist_identifier.py +35 -0
  205. wbportfolio/migrations/0045_alter_instrument_financial_instrument.py +26 -0
  206. wbportfolio/migrations/0046_add_product_default_account.py +166 -0
  207. wbportfolio/migrations/0047_remove_product_default_sub_account.py +14 -0
  208. wbportfolio/migrations/0048_alter_trade_status.py +29 -0
  209. wbportfolio/migrations/0049_trade_claimed_shares.py +25 -0
  210. wbportfolio/migrations/0050_fees_fee_date_fees_wbportfolio_transac_1f7a29_idx.py +44 -0
  211. wbportfolio/migrations/0051_delete_macroreview.py +11 -0
  212. wbportfolio/migrations/0052_remove_cash_instrument_ptr_and_more.py +888 -0
  213. wbportfolio/migrations/0053_remove_product_group.py +132 -0
  214. wbportfolio/migrations/0054_portfolioinstrumentpreferredclassificationthroughmodel_and_more.py +270 -0
  215. wbportfolio/migrations/0055_remove_product__custom_management_rebates_and_more.py +139 -0
  216. wbportfolio/migrations/0056_remove_companyportfoliodata_assets_under_management_currency_and_more.py +56 -0
  217. wbportfolio/migrations/0057_alter_portfolio_preferred_instrument_classifications_and_more.py +36 -0
  218. wbportfolio/migrations/0058_pmsinstrument.py +23 -0
  219. wbportfolio/migrations/0059_fees_unique_fees.py +51 -0
  220. wbportfolio/migrations/0060_alter_portfolioportfoliothroughmodel_type.py +21 -0
  221. wbportfolio/migrations/0061_portfolio_bank_accounts_product_bank_account_and_more.py +175 -0
  222. wbportfolio/migrations/0062_alter_dailyportfoliocashflow_options.py +20 -0
  223. wbportfolio/migrations/0063_accountreconciliation_accountreconciliationline_and_more.py +133 -0
  224. wbportfolio/migrations/0064_alter_portfolio_managers_portfolio_is_tracked_and_more.py +40 -0
  225. wbportfolio/migrations/0065_alter_portfolio_managers_claim_as_shares_and_more.py +73 -0
  226. wbportfolio/migrations/0066_assetposition_initial_shares_at_custodian_and_more.py +108 -0
  227. wbportfolio/migrations/0067_assetposition_unique_asset_position.py +77 -0
  228. wbportfolio/migrations/0068_trade_internal_trade_trade_marked_as_internal_and_more.py +59 -0
  229. wbportfolio/migrations/0069_remove_portfolio_is_invested_and_more.py +56 -0
  230. wbportfolio/migrations/0070_remove_assetposition_unique_asset_position_and_more.py +82 -0
  231. wbportfolio/migrations/0071_alter_trade_options_alter_trade_order.py +22 -0
  232. wbportfolio/migrations/__init__.py +0 -0
  233. wbportfolio/models/__init__.py +26 -0
  234. wbportfolio/models/adjustments.py +246 -0
  235. wbportfolio/models/asset.py +869 -0
  236. wbportfolio/models/custodians.py +101 -0
  237. wbportfolio/models/indexes.py +33 -0
  238. wbportfolio/models/mixins/__init__.py +0 -0
  239. wbportfolio/models/mixins/instruments.py +127 -0
  240. wbportfolio/models/mixins/liquidity_stress_test.py +1307 -0
  241. wbportfolio/models/portfolio.py +1039 -0
  242. wbportfolio/models/portfolio_cash_flow.py +167 -0
  243. wbportfolio/models/portfolio_cash_targets.py +46 -0
  244. wbportfolio/models/portfolio_relationship.py +135 -0
  245. wbportfolio/models/portfolio_swing_pricings.py +51 -0
  246. wbportfolio/models/product_groups.py +230 -0
  247. wbportfolio/models/products.py +569 -0
  248. wbportfolio/models/reconciliations/__init__.py +2 -0
  249. wbportfolio/models/reconciliations/account_reconciliation_lines.py +192 -0
  250. wbportfolio/models/reconciliations/account_reconciliations.py +102 -0
  251. wbportfolio/models/reconciliations/reconciliations.py +25 -0
  252. wbportfolio/models/registers.py +132 -0
  253. wbportfolio/models/roles.py +208 -0
  254. wbportfolio/models/synchronization/__init__.py +3 -0
  255. wbportfolio/models/synchronization/portfolio_synchronization.py +292 -0
  256. wbportfolio/models/synchronization/price_computation.py +200 -0
  257. wbportfolio/models/synchronization/synchronization.py +188 -0
  258. wbportfolio/models/transactions/__init__.py +7 -0
  259. wbportfolio/models/transactions/claim.py +634 -0
  260. wbportfolio/models/transactions/dividends.py +31 -0
  261. wbportfolio/models/transactions/expiry.py +7 -0
  262. wbportfolio/models/transactions/fees.py +153 -0
  263. wbportfolio/models/transactions/trade_proposals.py +502 -0
  264. wbportfolio/models/transactions/trades.py +704 -0
  265. wbportfolio/models/transactions/transactions.py +211 -0
  266. wbportfolio/models/utils.py +12 -0
  267. wbportfolio/permissions.py +13 -0
  268. wbportfolio/pms/__init__.py +0 -0
  269. wbportfolio/pms/statistics/__init__.py +0 -0
  270. wbportfolio/pms/trading/__init__.py +1 -0
  271. wbportfolio/pms/trading/handler.py +164 -0
  272. wbportfolio/pms/typing.py +194 -0
  273. wbportfolio/preferences.py +6 -0
  274. wbportfolio/reports/__init__.py +0 -0
  275. wbportfolio/reports/monthly_position_report.py +74 -0
  276. wbportfolio/risk_management/__init__.py +0 -0
  277. wbportfolio/risk_management/backends/__init__.py +11 -0
  278. wbportfolio/risk_management/backends/accounts.py +166 -0
  279. wbportfolio/risk_management/backends/controversy_portfolio.py +63 -0
  280. wbportfolio/risk_management/backends/exposure_portfolio.py +203 -0
  281. wbportfolio/risk_management/backends/instrument_list_portfolio.py +89 -0
  282. wbportfolio/risk_management/backends/liquidity_risk.py +86 -0
  283. wbportfolio/risk_management/backends/liquidity_stress_instrument.py +86 -0
  284. wbportfolio/risk_management/backends/mixins.py +220 -0
  285. wbportfolio/risk_management/backends/product_integrity.py +111 -0
  286. wbportfolio/risk_management/backends/stop_loss_instrument.py +24 -0
  287. wbportfolio/risk_management/backends/stop_loss_portfolio.py +36 -0
  288. wbportfolio/risk_management/backends/ucits_portfolio.py +63 -0
  289. wbportfolio/risk_management/tests/__init__.py +0 -0
  290. wbportfolio/risk_management/tests/conftest.py +15 -0
  291. wbportfolio/risk_management/tests/test_accounts.py +98 -0
  292. wbportfolio/risk_management/tests/test_controversy_portfolio.py +33 -0
  293. wbportfolio/risk_management/tests/test_exposure_portfolio.py +94 -0
  294. wbportfolio/risk_management/tests/test_instrument_list_portfolio.py +60 -0
  295. wbportfolio/risk_management/tests/test_liquidity_risk.py +47 -0
  296. wbportfolio/risk_management/tests/test_product_integrity.py +55 -0
  297. wbportfolio/risk_management/tests/test_stop_loss_instrument.py +110 -0
  298. wbportfolio/risk_management/tests/test_stop_loss_portfolio.py +119 -0
  299. wbportfolio/risk_management/tests/test_ucits_portfolio.py +39 -0
  300. wbportfolio/serializers/__init__.py +42 -0
  301. wbportfolio/serializers/adjustments.py +24 -0
  302. wbportfolio/serializers/assets.py +166 -0
  303. wbportfolio/serializers/custodians.py +26 -0
  304. wbportfolio/serializers/portfolio_cash_flow.py +48 -0
  305. wbportfolio/serializers/portfolio_cash_targets.py +20 -0
  306. wbportfolio/serializers/portfolio_relationship.py +53 -0
  307. wbportfolio/serializers/portfolio_swing_pricing.py +20 -0
  308. wbportfolio/serializers/portfolios.py +143 -0
  309. wbportfolio/serializers/positions.py +76 -0
  310. wbportfolio/serializers/product_group.py +88 -0
  311. wbportfolio/serializers/products.py +331 -0
  312. wbportfolio/serializers/reconciliations.py +171 -0
  313. wbportfolio/serializers/registers.py +72 -0
  314. wbportfolio/serializers/roles.py +60 -0
  315. wbportfolio/serializers/signals.py +157 -0
  316. wbportfolio/serializers/synchronization.py +18 -0
  317. wbportfolio/serializers/transactions/__init__.py +24 -0
  318. wbportfolio/serializers/transactions/claim.py +310 -0
  319. wbportfolio/serializers/transactions/dividends.py +18 -0
  320. wbportfolio/serializers/transactions/expiry.py +18 -0
  321. wbportfolio/serializers/transactions/fees.py +32 -0
  322. wbportfolio/serializers/transactions/trades.py +315 -0
  323. wbportfolio/serializers/transactions/transactions.py +84 -0
  324. wbportfolio/tasks.py +125 -0
  325. wbportfolio/tests/__init__.py +0 -0
  326. wbportfolio/tests/conftest.py +164 -0
  327. wbportfolio/tests/models/__init__.py +0 -0
  328. wbportfolio/tests/models/test_account_reconciliation.py +191 -0
  329. wbportfolio/tests/models/test_assets.py +193 -0
  330. wbportfolio/tests/models/test_custodians.py +12 -0
  331. wbportfolio/tests/models/test_customer_trades.py +113 -0
  332. wbportfolio/tests/models/test_dividends.py +7 -0
  333. wbportfolio/tests/models/test_imports.py +192 -0
  334. wbportfolio/tests/models/test_instrument_mixins.py +48 -0
  335. wbportfolio/tests/models/test_merge.py +133 -0
  336. wbportfolio/tests/models/test_portfolio_cash_flow.py +112 -0
  337. wbportfolio/tests/models/test_portfolio_cash_targets.py +27 -0
  338. wbportfolio/tests/models/test_portfolio_swing_pricings.py +42 -0
  339. wbportfolio/tests/models/test_portfolios.py +676 -0
  340. wbportfolio/tests/models/test_product_groups.py +80 -0
  341. wbportfolio/tests/models/test_products.py +187 -0
  342. wbportfolio/tests/models/test_roles.py +82 -0
  343. wbportfolio/tests/models/test_splits.py +233 -0
  344. wbportfolio/tests/models/test_synchronization.py +617 -0
  345. wbportfolio/tests/models/transactions/__init__.py +0 -0
  346. wbportfolio/tests/models/transactions/test_claim.py +129 -0
  347. wbportfolio/tests/models/transactions/test_fees.py +65 -0
  348. wbportfolio/tests/models/transactions/test_trades.py +204 -0
  349. wbportfolio/tests/models/utils.py +13 -0
  350. wbportfolio/tests/serializers/__init__.py +0 -0
  351. wbportfolio/tests/serializers/test_claims.py +21 -0
  352. wbportfolio/tests/signals.py +151 -0
  353. wbportfolio/tests/tests.py +31 -0
  354. wbportfolio/tests/viewsets/__init__.py +0 -0
  355. wbportfolio/tests/viewsets/test_assets.py +67 -0
  356. wbportfolio/tests/viewsets/test_performances.py +72 -0
  357. wbportfolio/tests/viewsets/test_products.py +92 -0
  358. wbportfolio/tests/viewsets/transactions/__init__.py +0 -0
  359. wbportfolio/tests/viewsets/transactions/test_claims.py +146 -0
  360. wbportfolio/urls.py +247 -0
  361. wbportfolio/utils.py +30 -0
  362. wbportfolio/viewsets/__init__.py +57 -0
  363. wbportfolio/viewsets/adjustments.py +46 -0
  364. wbportfolio/viewsets/assets.py +562 -0
  365. wbportfolio/viewsets/assets_and_net_new_money_progression.py +117 -0
  366. wbportfolio/viewsets/charts/__init__.py +1 -0
  367. wbportfolio/viewsets/charts/assets.py +247 -0
  368. wbportfolio/viewsets/configs/__init__.py +6 -0
  369. wbportfolio/viewsets/configs/buttons/__init__.py +23 -0
  370. wbportfolio/viewsets/configs/buttons/adjustments.py +13 -0
  371. wbportfolio/viewsets/configs/buttons/assets.py +145 -0
  372. wbportfolio/viewsets/configs/buttons/claims.py +83 -0
  373. wbportfolio/viewsets/configs/buttons/custodians.py +76 -0
  374. wbportfolio/viewsets/configs/buttons/fees.py +14 -0
  375. wbportfolio/viewsets/configs/buttons/mixins.py +88 -0
  376. wbportfolio/viewsets/configs/buttons/portfolios.py +115 -0
  377. wbportfolio/viewsets/configs/buttons/products.py +41 -0
  378. wbportfolio/viewsets/configs/buttons/reconciliations.py +65 -0
  379. wbportfolio/viewsets/configs/buttons/registers.py +11 -0
  380. wbportfolio/viewsets/configs/buttons/signals.py +68 -0
  381. wbportfolio/viewsets/configs/buttons/trade_proposals.py +25 -0
  382. wbportfolio/viewsets/configs/buttons/trades.py +144 -0
  383. wbportfolio/viewsets/configs/display/__init__.py +61 -0
  384. wbportfolio/viewsets/configs/display/adjustments.py +81 -0
  385. wbportfolio/viewsets/configs/display/assets.py +265 -0
  386. wbportfolio/viewsets/configs/display/claim.py +299 -0
  387. wbportfolio/viewsets/configs/display/custodians.py +24 -0
  388. wbportfolio/viewsets/configs/display/esg.py +88 -0
  389. wbportfolio/viewsets/configs/display/fees.py +133 -0
  390. wbportfolio/viewsets/configs/display/portfolio_cash_flow.py +103 -0
  391. wbportfolio/viewsets/configs/display/portfolio_relationship.py +38 -0
  392. wbportfolio/viewsets/configs/display/portfolios.py +125 -0
  393. wbportfolio/viewsets/configs/display/positions.py +75 -0
  394. wbportfolio/viewsets/configs/display/product_groups.py +54 -0
  395. wbportfolio/viewsets/configs/display/product_performance.py +241 -0
  396. wbportfolio/viewsets/configs/display/products.py +249 -0
  397. wbportfolio/viewsets/configs/display/reconciliations.py +151 -0
  398. wbportfolio/viewsets/configs/display/registers.py +71 -0
  399. wbportfolio/viewsets/configs/display/roles.py +49 -0
  400. wbportfolio/viewsets/configs/display/trade_proposals.py +97 -0
  401. wbportfolio/viewsets/configs/display/trades.py +359 -0
  402. wbportfolio/viewsets/configs/display/transactions.py +55 -0
  403. wbportfolio/viewsets/configs/endpoints/__init__.py +75 -0
  404. wbportfolio/viewsets/configs/endpoints/adjustments.py +17 -0
  405. wbportfolio/viewsets/configs/endpoints/assets.py +115 -0
  406. wbportfolio/viewsets/configs/endpoints/claim.py +106 -0
  407. wbportfolio/viewsets/configs/endpoints/custodians.py +6 -0
  408. wbportfolio/viewsets/configs/endpoints/esg.py +14 -0
  409. wbportfolio/viewsets/configs/endpoints/fees.py +26 -0
  410. wbportfolio/viewsets/configs/endpoints/portfolio_relationship.py +23 -0
  411. wbportfolio/viewsets/configs/endpoints/portfolios.py +43 -0
  412. wbportfolio/viewsets/configs/endpoints/positions.py +18 -0
  413. wbportfolio/viewsets/configs/endpoints/product_groups.py +11 -0
  414. wbportfolio/viewsets/configs/endpoints/product_performance.py +29 -0
  415. wbportfolio/viewsets/configs/endpoints/products.py +37 -0
  416. wbportfolio/viewsets/configs/endpoints/reconciliations.py +31 -0
  417. wbportfolio/viewsets/configs/endpoints/roles.py +9 -0
  418. wbportfolio/viewsets/configs/endpoints/trade_proposals.py +17 -0
  419. wbportfolio/viewsets/configs/endpoints/trades.py +82 -0
  420. wbportfolio/viewsets/configs/endpoints/transactions.py +17 -0
  421. wbportfolio/viewsets/configs/menu/__init__.py +30 -0
  422. wbportfolio/viewsets/configs/menu/adjustments.py +8 -0
  423. wbportfolio/viewsets/configs/menu/assets.py +8 -0
  424. wbportfolio/viewsets/configs/menu/claim.py +41 -0
  425. wbportfolio/viewsets/configs/menu/custodians.py +11 -0
  426. wbportfolio/viewsets/configs/menu/fees.py +13 -0
  427. wbportfolio/viewsets/configs/menu/instrument_prices.py +10 -0
  428. wbportfolio/viewsets/configs/menu/portfolio_cash_flow.py +8 -0
  429. wbportfolio/viewsets/configs/menu/portfolios.py +15 -0
  430. wbportfolio/viewsets/configs/menu/positions.py +14 -0
  431. wbportfolio/viewsets/configs/menu/product_groups.py +10 -0
  432. wbportfolio/viewsets/configs/menu/product_performance.py +25 -0
  433. wbportfolio/viewsets/configs/menu/products.py +15 -0
  434. wbportfolio/viewsets/configs/menu/reconciliations.py +7 -0
  435. wbportfolio/viewsets/configs/menu/registers.py +10 -0
  436. wbportfolio/viewsets/configs/menu/roles.py +16 -0
  437. wbportfolio/viewsets/configs/menu/trades.py +18 -0
  438. wbportfolio/viewsets/configs/menu/transactions.py +8 -0
  439. wbportfolio/viewsets/configs/previews/__init__.py +1 -0
  440. wbportfolio/viewsets/configs/previews/portfolios.py +21 -0
  441. wbportfolio/viewsets/configs/titles/__init__.py +65 -0
  442. wbportfolio/viewsets/configs/titles/adjustments.py +19 -0
  443. wbportfolio/viewsets/configs/titles/assets.py +57 -0
  444. wbportfolio/viewsets/configs/titles/assets_and_net_new_money_progression.py +6 -0
  445. wbportfolio/viewsets/configs/titles/claim.py +81 -0
  446. wbportfolio/viewsets/configs/titles/custodians.py +12 -0
  447. wbportfolio/viewsets/configs/titles/esg.py +10 -0
  448. wbportfolio/viewsets/configs/titles/fees.py +25 -0
  449. wbportfolio/viewsets/configs/titles/instrument_prices.py +20 -0
  450. wbportfolio/viewsets/configs/titles/portfolios.py +32 -0
  451. wbportfolio/viewsets/configs/titles/positions.py +11 -0
  452. wbportfolio/viewsets/configs/titles/product_groups.py +12 -0
  453. wbportfolio/viewsets/configs/titles/product_performance.py +16 -0
  454. wbportfolio/viewsets/configs/titles/products.py +6 -0
  455. wbportfolio/viewsets/configs/titles/registers.py +12 -0
  456. wbportfolio/viewsets/configs/titles/roles.py +23 -0
  457. wbportfolio/viewsets/configs/titles/trades.py +51 -0
  458. wbportfolio/viewsets/configs/titles/transactions.py +8 -0
  459. wbportfolio/viewsets/custodians.py +66 -0
  460. wbportfolio/viewsets/esg.py +165 -0
  461. wbportfolio/viewsets/mixins.py +48 -0
  462. wbportfolio/viewsets/portfolio_cash_flow.py +31 -0
  463. wbportfolio/viewsets/portfolio_cash_targets.py +8 -0
  464. wbportfolio/viewsets/portfolio_relationship.py +46 -0
  465. wbportfolio/viewsets/portfolio_swing_pricing.py +8 -0
  466. wbportfolio/viewsets/portfolios.py +154 -0
  467. wbportfolio/viewsets/positions.py +292 -0
  468. wbportfolio/viewsets/product_groups.py +84 -0
  469. wbportfolio/viewsets/product_performance.py +646 -0
  470. wbportfolio/viewsets/products.py +529 -0
  471. wbportfolio/viewsets/reconciliations.py +160 -0
  472. wbportfolio/viewsets/registers.py +75 -0
  473. wbportfolio/viewsets/roles.py +44 -0
  474. wbportfolio/viewsets/signals.py +42 -0
  475. wbportfolio/viewsets/synchronization.py +25 -0
  476. wbportfolio/viewsets/transactions/__init__.py +40 -0
  477. wbportfolio/viewsets/transactions/claim.py +933 -0
  478. wbportfolio/viewsets/transactions/fees.py +190 -0
  479. wbportfolio/viewsets/transactions/mixins.py +19 -0
  480. wbportfolio/viewsets/transactions/trade_proposals.py +93 -0
  481. wbportfolio/viewsets/transactions/trades.py +395 -0
  482. wbportfolio/viewsets/transactions/transactions.py +123 -0
  483. wbportfolio-2.2.1.dist-info/METADATA +21 -0
  484. wbportfolio-2.2.1.dist-info/RECORD +486 -0
  485. wbportfolio-2.2.1.dist-info/WHEEL +5 -0
  486. wbportfolio-2.2.1.dist-info/licenses/LICENSE +4 -0
@@ -0,0 +1,41 @@
1
+ import rstr
2
+ from faker import Faker
3
+ from import_export import fields
4
+ from import_export.widgets import ForeignKeyWidget
5
+ from wbcore.contrib.io.resources import FilterModelResource
6
+ from wbfdm.models import Instrument
7
+ from wbportfolio.models import Trade
8
+
9
+ fake = Faker()
10
+
11
+
12
+ class TradeProposalTradeResource(FilterModelResource):
13
+ """
14
+ Trade Resource class to use to import trade from the trade proposal
15
+ """
16
+
17
+ DUMMY_FIELD_MAP = {
18
+ "underlying_instrument": lambda: rstr.xeger("([A-Z]{2}[A-Z0-9]{9}[0-9]{1})"),
19
+ "weighting": 1.0,
20
+ "shares": 1000.2536,
21
+ "comment": lambda: fake.sentence(),
22
+ "order": 1,
23
+ }
24
+ underlying_instrument = fields.Field(
25
+ column_name="underlying_instrument",
26
+ attribute="underlying_instrument",
27
+ widget=ForeignKeyWidget(Instrument, field="isin"),
28
+ )
29
+
30
+ class Meta:
31
+ import_id_fields = ("id",)
32
+ fields = (
33
+ "id",
34
+ "underlying_instrument",
35
+ "weighting",
36
+ "shares",
37
+ "comment",
38
+ "order",
39
+ )
40
+ export_order = fields
41
+ model = Trade
@@ -0,0 +1,42 @@
1
+ import os
2
+
3
+ import dateparser
4
+ from xlrd.xldate import xldate_as_datetime
5
+
6
+
7
+ def get_fields_name(model):
8
+ return [field.name for field in model._meta.get_fields()]
9
+
10
+
11
+ def get_file_extension(file):
12
+ name, extension = os.path.splitext(file.name)
13
+ return extension.lower()
14
+
15
+
16
+ def convert_string_to_number(string):
17
+ if string == "nan" or not string:
18
+ return 0.0
19
+ if type(string) in (int, float):
20
+ return float(string)
21
+ try:
22
+ return float(string.replace(" ", "").replace(",", ""))
23
+ except ValueError:
24
+ return 0.0
25
+
26
+
27
+ def parse_date(date, formats=[]):
28
+ if isinstance(date, int) or isinstance(date, float):
29
+ return xldate_as_datetime(int(date), 0).date()
30
+ if isinstance(date, str):
31
+ if format:
32
+ return dateparser.parse(date, date_formats=formats).date()
33
+ else:
34
+ return dateparser.parse(date).date()
35
+
36
+
37
+ def extract_exchange_ticker(bbg_ticker):
38
+ try:
39
+ ticker, exchange = bbg_ticker.split(" ")[0:2]
40
+ except Exception:
41
+ ticker, exchange = bbg_ticker, None
42
+ return ticker, exchange
File without changes
@@ -0,0 +1,2 @@
1
+ from .portfolio_base import PortfolioBaseMetricBackend
2
+ from .portfolio_esg import PortfolioESGMetricBackend
@@ -0,0 +1,86 @@
1
+ from datetime import date
2
+
3
+ import numpy as np
4
+ import pandas as pd
5
+ from django.db.models import Exists, OuterRef, QuerySet
6
+ from wbfdm.contrib.metric.backends.base import AbstractBackend, Metric
7
+ from wbfdm.contrib.metric.exceptions import MetricInvalidParameterException
8
+ from wbfdm.models import InstrumentType
9
+ from wbportfolio.models import AssetPosition, InstrumentPortfolioThroughModel, Portfolio
10
+
11
+
12
+ class PortfolioMetricBaseBackend(AbstractBackend[Portfolio]):
13
+ BASKET_MODEL_CLASS = Portfolio
14
+
15
+ def convert_df_into_metrics(
16
+ self,
17
+ df: pd.DataFrame | pd.Series,
18
+ key: str,
19
+ portfolio_id: int,
20
+ metric_date: date,
21
+ columns_map: dict[int | str, str] | None = None,
22
+ dependency_metrics: list["Metric"] | None = None,
23
+ ) -> list[Metric]:
24
+ if not dependency_metrics:
25
+ dependency_metrics = []
26
+ metrics = []
27
+ df = df.replace([np.inf, -np.inf, np.nan], None)
28
+ if isinstance(df, pd.DataFrame):
29
+ if columns_map:
30
+ df = df.rename(columns=columns_map)
31
+ for instrument_id, metric in df.to_dict("index").items():
32
+ metrics.append(
33
+ Metric(
34
+ basket_id=portfolio_id,
35
+ basket_content_type_id=self.content_type.id,
36
+ key=key,
37
+ metrics=metric,
38
+ date=metric_date,
39
+ instrument_id=instrument_id,
40
+ dependency_metrics=dependency_metrics,
41
+ )
42
+ )
43
+ else:
44
+ if columns_map:
45
+ df = df.rename(index=columns_map)
46
+ metrics.append(
47
+ Metric(
48
+ basket_id=portfolio_id,
49
+ basket_content_type_id=self.content_type.id,
50
+ key=key,
51
+ metrics=df.to_dict(),
52
+ date=metric_date,
53
+ dependency_metrics=dependency_metrics,
54
+ )
55
+ )
56
+ return metrics
57
+
58
+ def _get_valid_date(self, portfolio):
59
+ qs = portfolio.assets.filter(is_estimated=False)
60
+ if self.val_date is not None:
61
+ qs = qs.filter(date__lte=self.val_date)
62
+ try:
63
+ return qs.latest("date").date
64
+ except AssetPosition.DoesNotExist:
65
+ raise MetricInvalidParameterException()
66
+
67
+ def get_queryset(self) -> QuerySet[Portfolio]:
68
+ product_portfolios = (
69
+ super()
70
+ .get_queryset()
71
+ .annotate(
72
+ has_product=Exists(
73
+ InstrumentPortfolioThroughModel.objects.filter(
74
+ instrument__instrument_type=InstrumentType.PRODUCT, portfolio=OuterRef("pk")
75
+ )
76
+ )
77
+ )
78
+ .filter(has_product=True, is_active=True)
79
+ )
80
+ try:
81
+ last_position_date = (
82
+ AssetPosition.objects.filter(portfolio__in=product_portfolios, is_estimated=False).latest("date").date
83
+ )
84
+ except AssetPosition.DoesNotExist:
85
+ last_position_date = date.today()
86
+ return product_portfolios.filter_invested_at_date(last_position_date)
@@ -0,0 +1,222 @@
1
+ from wbfdm.contrib.metric.dto import MetricField, MetricKey
2
+
3
+ PORTFOLIO_EBIT = MetricKey(
4
+ key="portfolio_ebit",
5
+ label="EBIT",
6
+ subfields=[
7
+ MetricField(
8
+ key="fy_3",
9
+ label="FY-3",
10
+ list_display_kwargs={"show": "open"},
11
+ decorators=[{"position": "left", "value": "$"}],
12
+ ),
13
+ MetricField(
14
+ key="fy_2",
15
+ label="FY-2",
16
+ list_display_kwargs={"show": "open"},
17
+ decorators=[{"position": "left", "value": "$"}],
18
+ ),
19
+ MetricField(
20
+ key="fy_1",
21
+ label="FY-1",
22
+ list_display_kwargs={"show": "open"},
23
+ decorators=[{"position": "left", "value": "$"}],
24
+ ),
25
+ MetricField(
26
+ key="fy0",
27
+ label="FY0",
28
+ list_display_kwargs={"show": "open"},
29
+ decorators=[{"position": "left", "value": "$"}],
30
+ ),
31
+ MetricField(key="fy1", label="FY1", decorators=[{"position": "left", "value": "$"}]),
32
+ MetricField(
33
+ key="fy2",
34
+ label="FY2",
35
+ list_display_kwargs={"show": "open"},
36
+ decorators=[{"position": "left", "value": "$"}],
37
+ ),
38
+ MetricField(
39
+ key="fy3",
40
+ label="FY3",
41
+ list_display_kwargs={"show": "open"},
42
+ decorators=[{"position": "left", "value": "$"}],
43
+ ),
44
+ MetricField(
45
+ key="fy4",
46
+ label="FY4",
47
+ list_display_kwargs={"show": "open"},
48
+ decorators=[{"position": "left", "value": "$"}],
49
+ ),
50
+ ],
51
+ )
52
+
53
+ PORTFOLIO_TOTAL_ASSETS = MetricKey(
54
+ key="portfolio_total_assets",
55
+ label="Total Assets",
56
+ subfields=[
57
+ MetricField(
58
+ key="fy_3",
59
+ label="FY-3",
60
+ list_display_kwargs={"show": "open"},
61
+ decorators=[{"position": "left", "value": "$"}],
62
+ ),
63
+ MetricField(
64
+ key="fy_2",
65
+ label="FY-2",
66
+ list_display_kwargs={"show": "open"},
67
+ decorators=[{"position": "left", "value": "$"}],
68
+ ),
69
+ MetricField(
70
+ key="fy_1",
71
+ label="FY-1",
72
+ list_display_kwargs={"show": "open"},
73
+ decorators=[{"position": "left", "value": "$"}],
74
+ ),
75
+ MetricField(
76
+ key="fy0",
77
+ label="FY0",
78
+ list_display_kwargs={"show": "open"},
79
+ decorators=[{"position": "left", "value": "$"}],
80
+ ),
81
+ MetricField(key="fy1", label="FY1", decorators=[{"position": "left", "value": "$"}]),
82
+ MetricField(
83
+ key="fy2",
84
+ label="FY2",
85
+ list_display_kwargs={"show": "open"},
86
+ decorators=[{"position": "left", "value": "$"}],
87
+ ),
88
+ MetricField(
89
+ key="fy3",
90
+ label="FY3",
91
+ list_display_kwargs={"show": "open"},
92
+ decorators=[{"position": "left", "value": "$"}],
93
+ ),
94
+ MetricField(
95
+ key="fy4",
96
+ label="FY4",
97
+ list_display_kwargs={"show": "open"},
98
+ decorators=[{"position": "left", "value": "$"}],
99
+ ),
100
+ ],
101
+ )
102
+
103
+ PORTFOLIO_LIABILITIES = MetricKey(
104
+ key="portfolio_liabilities",
105
+ label="Liabilities",
106
+ subfields=[
107
+ MetricField(
108
+ key="fy_3",
109
+ label="FY-3",
110
+ list_display_kwargs={"show": "open"},
111
+ decorators=[{"position": "left", "value": "$"}],
112
+ ),
113
+ MetricField(
114
+ key="fy_2",
115
+ label="FY-2",
116
+ list_display_kwargs={"show": "open"},
117
+ decorators=[{"position": "left", "value": "$"}],
118
+ ),
119
+ MetricField(
120
+ key="fy_1",
121
+ label="FY-1",
122
+ list_display_kwargs={"show": "open"},
123
+ decorators=[{"position": "left", "value": "$"}],
124
+ ),
125
+ MetricField(
126
+ key="fy0",
127
+ label="FY0",
128
+ list_display_kwargs={"show": "open"},
129
+ decorators=[{"position": "left", "value": "$"}],
130
+ ),
131
+ MetricField(key="fy1", label="FY1", decorators=[{"position": "left", "value": "$"}]),
132
+ MetricField(
133
+ key="fy2",
134
+ label="FY2",
135
+ list_display_kwargs={"show": "open"},
136
+ decorators=[{"position": "left", "value": "$"}],
137
+ ),
138
+ MetricField(
139
+ key="fy3",
140
+ label="FY3",
141
+ list_display_kwargs={"show": "open"},
142
+ decorators=[{"position": "left", "value": "$"}],
143
+ ),
144
+ MetricField(
145
+ key="fy4",
146
+ label="FY4",
147
+ list_display_kwargs={"show": "open"},
148
+ decorators=[{"position": "left", "value": "$"}],
149
+ ),
150
+ ],
151
+ )
152
+
153
+ PORTFOLIO_CAPITAL_EMPLOYED = MetricKey(
154
+ key="portfolio_capital_employed",
155
+ label="Capital Employed",
156
+ subfields=[
157
+ MetricField(
158
+ key="fy_3",
159
+ label="FY-3",
160
+ list_display_kwargs={"show": "open"},
161
+ decorators=[{"position": "left", "value": "$"}],
162
+ ),
163
+ MetricField(
164
+ key="fy_2",
165
+ label="FY-2",
166
+ list_display_kwargs={"show": "open"},
167
+ decorators=[{"position": "left", "value": "$"}],
168
+ ),
169
+ MetricField(
170
+ key="fy_1",
171
+ label="FY-1",
172
+ list_display_kwargs={"show": "open"},
173
+ decorators=[{"position": "left", "value": "$"}],
174
+ ),
175
+ MetricField(
176
+ key="fy0",
177
+ label="FY0",
178
+ list_display_kwargs={"show": "open"},
179
+ decorators=[{"position": "left", "value": "$"}],
180
+ ),
181
+ MetricField(key="fy1", label="FY1", decorators=[{"position": "left", "value": "$"}]),
182
+ MetricField(
183
+ key="fy2",
184
+ label="FY2",
185
+ list_display_kwargs={"show": "open"},
186
+ decorators=[{"position": "left", "value": "$"}],
187
+ ),
188
+ MetricField(
189
+ key="fy3",
190
+ label="FY3",
191
+ list_display_kwargs={"show": "open"},
192
+ decorators=[{"position": "left", "value": "$"}],
193
+ ),
194
+ MetricField(
195
+ key="fy4",
196
+ label="FY4",
197
+ list_display_kwargs={"show": "open"},
198
+ decorators=[{"position": "left", "value": "$"}],
199
+ ),
200
+ ],
201
+ )
202
+
203
+ PORTFOLIO_ROCE = MetricKey(
204
+ key="portfolio_roce",
205
+ label="ROCE",
206
+ subfields=[
207
+ MetricField(
208
+ key="fy_3", label="FY-3", list_display_kwargs={"show": "open"}, serializer_kwargs={"percent": True}
209
+ ),
210
+ MetricField(
211
+ key="fy_2", label="FY-2", list_display_kwargs={"show": "open"}, serializer_kwargs={"percent": True}
212
+ ),
213
+ MetricField(
214
+ key="fy_1", label="FY-1", list_display_kwargs={"show": "open"}, serializer_kwargs={"percent": True}
215
+ ),
216
+ MetricField(key="fy0", label="FY0", list_display_kwargs={"show": "open"}, serializer_kwargs={"percent": True}),
217
+ MetricField(key="fy1", label="FY1", serializer_kwargs={"percent": True}),
218
+ MetricField(key="fy2", label="FY2", list_display_kwargs={"show": "open"}, serializer_kwargs={"percent": True}),
219
+ MetricField(key="fy3", label="FY3", list_display_kwargs={"show": "open"}, serializer_kwargs={"percent": True}),
220
+ MetricField(key="fy4", label="FY4", list_display_kwargs={"show": "open"}, serializer_kwargs={"percent": True}),
221
+ ],
222
+ )
@@ -0,0 +1,255 @@
1
+ from contextlib import suppress
2
+ from datetime import date
3
+ from typing import Generator
4
+
5
+ import pandas as pd
6
+ from django.db.models import Q
7
+ from django.utils.functional import cached_property
8
+ from wbfdm.contrib.metric.backends.base import Metric
9
+ from wbfdm.contrib.metric.decorators import register
10
+ from wbfdm.enums import Financial, MarketData
11
+ from wbfdm.models import Instrument
12
+ from wbportfolio.models import Portfolio
13
+
14
+ from .base import PortfolioMetricBaseBackend
15
+ from .constants import (
16
+ PORTFOLIO_CAPITAL_EMPLOYED,
17
+ PORTFOLIO_EBIT,
18
+ PORTFOLIO_LIABILITIES,
19
+ PORTFOLIO_ROCE,
20
+ PORTFOLIO_TOTAL_ASSETS,
21
+ )
22
+
23
+
24
+ class DataLoader:
25
+ def __init__(
26
+ self, portfolio: Portfolio, val_date: date, past_offset: int = 4, forecast_offset: int = 4, freq: str = "YE"
27
+ ):
28
+ self.portfolio = portfolio
29
+ self.val_date = val_date
30
+ self.instruments = Instrument.objects.filter(id__in=self.base_df.index)
31
+ self.past_offset = past_offset
32
+ self.forecast_offset = forecast_offset
33
+ self.pivot_dates = list(
34
+ pd.date_range(end=self.val_date, periods=self.past_offset, freq=freq, inclusive="left")
35
+ ) + list(pd.date_range(start=self.val_date, periods=self.past_offset, freq=freq, inclusive="right"))
36
+ self.columns_map = {
37
+ self.val_date.year + offset: f"fy{offset + 1}".replace("-", "_")
38
+ for offset in range(-1 * self.past_offset, self.forecast_offset)
39
+ }
40
+ self.from_year = self.val_date.year - self.past_offset
41
+ self.to_year = self.val_date.year + self.forecast_offset - 1
42
+
43
+ self.empty_series = pd.Series(dtype="float64")
44
+
45
+ def _get_financial_df(self, financial: Financial, target_currency: str = "USD", **kwargs) -> pd.DataFrame:
46
+ """
47
+ Private helper method to gather necessary financial data from the finanical dataloader and pivot it into Instrument*Years format
48
+
49
+ Args:
50
+ financial: Financial Metric
51
+ target_currency: Target currency
52
+ **kwargs: fdm datalaoder keyword argument
53
+
54
+ Returns:
55
+ A dataframe whose index is the instrument ids and columns are the years offsets
56
+ """
57
+
58
+ df = pd.DataFrame(
59
+ self.instruments.dl.financials(
60
+ values=[financial],
61
+ from_year=self.from_year,
62
+ to_year=self.to_year,
63
+ target_currency=target_currency,
64
+ **kwargs,
65
+ )
66
+ )
67
+ if df.empty:
68
+ raise ValueError(f"No financial data for {financial.value}")
69
+ return df.pivot_table(index="instrument_id", values="value", columns="year", aggfunc="first")
70
+
71
+ # Data Input
72
+ @cached_property
73
+ def market_capitalization_df(self) -> pd.DataFrame:
74
+ df_list = []
75
+ for pivot_date in self.pivot_dates:
76
+ try:
77
+ mkt_caps = pd.DataFrame(
78
+ self.instruments.dl.market_data(
79
+ values=[MarketData.MARKET_CAPITALIZATION],
80
+ from_date=(pivot_date - pd.tseries.offsets.BDay(3)).date(),
81
+ to_date=pivot_date,
82
+ target_currency="USD",
83
+ )
84
+ )
85
+ res = (
86
+ mkt_caps[["instrument_id", "market_capitalization"]]
87
+ .groupby("instrument_id")
88
+ .last()["market_capitalization"]
89
+ ).astype(float)
90
+ except KeyError:
91
+ res = self.empty_series
92
+ res = res.rename(pivot_date.year)
93
+ df_list.append(res)
94
+ return pd.concat(df_list, axis=1).ffill(axis=1)
95
+
96
+ @cached_property
97
+ def ebit_usd_df(self) -> pd.DataFrame:
98
+ return self._get_financial_df(Financial.EBIT)
99
+
100
+ @cached_property
101
+ def total_assets_usd_df(self) -> pd.DataFrame:
102
+ return self._get_financial_df(Financial.TOTAL_ASSETS)
103
+
104
+ @cached_property
105
+ def liabilities_usd_df(self) -> pd.DataFrame:
106
+ return self._get_financial_df(Financial.CURRENT_LIABILITIES)
107
+
108
+ # Generate basic portfolio data as a dataframe
109
+ @cached_property
110
+ def base_df(self) -> pd.DataFrame:
111
+ df = pd.DataFrame(
112
+ self.portfolio.assets.filter(date=self.val_date)
113
+ .exclude(Q(underlying_instrument__is_cash=True) | Q(underlying_instrument__is_cash_equivalent=True))
114
+ .values("underlying_instrument", "shares", "price_fx_usd", "total_value_fx_usd", "weighting"),
115
+ columns=["underlying_instrument", "shares", "price_fx_usd", "total_value_fx_usd", "weighting"],
116
+ )
117
+ df = (
118
+ df.groupby("underlying_instrument")
119
+ .agg(
120
+ {
121
+ "shares": "sum",
122
+ "price_fx_usd": "first",
123
+ "total_value_fx_usd": "sum",
124
+ "weighting": "sum",
125
+ }
126
+ )
127
+ .astype(float)
128
+ )
129
+ df["weighting"] = df["weighting"] / df["weighting"].sum()
130
+ df.index = df.index.rename("instrument_id")
131
+ return df
132
+
133
+ # Portfolio Metrics
134
+
135
+ @cached_property
136
+ def portfolio_ownership_df(self) -> pd.DataFrame:
137
+ return (1.0 / self.market_capitalization_df).multiply(self.base_df.total_value_fx_usd, axis=0)
138
+
139
+ @cached_property
140
+ def portfolio_ebit_usd_df(self) -> pd.DataFrame:
141
+ df = self.portfolio_ownership_df * self.ebit_usd_df
142
+ df.aggregate = df.sum(axis=0)
143
+ return df
144
+
145
+ @cached_property
146
+ def portfolio_total_assets_usd_df(self) -> pd.DataFrame:
147
+ df = self.portfolio_ownership_df * self.total_assets_usd_df
148
+ df.aggregate = df.sum(axis=0)
149
+ return df
150
+
151
+ @cached_property
152
+ def portfolio_liabilities_usd_df(self) -> pd.DataFrame:
153
+ df = self.portfolio_ownership_df * self.liabilities_usd_df
154
+ df.aggregate = df.sum(axis=0)
155
+ return df
156
+
157
+ @cached_property
158
+ def portfolio_aggregation_capital_employed(self) -> pd.Series:
159
+ return self.portfolio_total_assets_usd_df.aggregate - self.portfolio_liabilities_usd_df.aggregate
160
+
161
+ @cached_property
162
+ def portfolio_aggregation_roce(self) -> pd.Series:
163
+ return self.portfolio_ebit_usd_df.aggregate / self.portfolio_aggregation_capital_employed.rolling(2).mean()
164
+
165
+
166
+ @register()
167
+ class PortfolioBaseMetricBackend(PortfolioMetricBaseBackend):
168
+ portfolio_ebit = PORTFOLIO_EBIT
169
+ portfolio_total_assets = PORTFOLIO_TOTAL_ASSETS
170
+ portfolio_liabilities = PORTFOLIO_LIABILITIES
171
+ portfolio_capital_employed = PORTFOLIO_CAPITAL_EMPLOYED
172
+ portfolio_roce = PORTFOLIO_ROCE
173
+
174
+ keys = [
175
+ PORTFOLIO_EBIT,
176
+ PORTFOLIO_TOTAL_ASSETS,
177
+ PORTFOLIO_LIABILITIES,
178
+ PORTFOLIO_CAPITAL_EMPLOYED,
179
+ PORTFOLIO_ROCE,
180
+ ]
181
+
182
+ def compute_metrics(self, basket: Portfolio) -> Generator[Metric, None, None]:
183
+ val_date = self._get_valid_date(basket)
184
+ dataloader = DataLoader(basket, val_date)
185
+
186
+ # Load all necessary metrics into a Metric DTO from the dataloader
187
+ with suppress(ValueError):
188
+ portfolio_ebit_metrics = self.convert_df_into_metrics(
189
+ dataloader.portfolio_ebit_usd_df,
190
+ "portfolio_ebit",
191
+ dataloader.portfolio.id,
192
+ val_date,
193
+ columns_map=dataloader.columns_map,
194
+ )
195
+ portfolio_aggregation_ebit_metrics = self.convert_df_into_metrics(
196
+ dataloader.portfolio_ebit_usd_df.aggregate,
197
+ "portfolio_ebit",
198
+ dataloader.portfolio.id,
199
+ val_date,
200
+ columns_map=dataloader.columns_map,
201
+ dependency_metrics=portfolio_ebit_metrics,
202
+ )
203
+
204
+ portfolio_total_assets_metrics = self.convert_df_into_metrics(
205
+ dataloader.portfolio_total_assets_usd_df,
206
+ "portfolio_total_assets",
207
+ dataloader.portfolio.id,
208
+ val_date,
209
+ columns_map=dataloader.columns_map,
210
+ )
211
+ portfolio_aggregation_total_assets_metrics = self.convert_df_into_metrics(
212
+ dataloader.portfolio_total_assets_usd_df.aggregate,
213
+ "portfolio_total_assets",
214
+ dataloader.portfolio.id,
215
+ val_date,
216
+ columns_map=dataloader.columns_map,
217
+ dependency_metrics=portfolio_total_assets_metrics,
218
+ )
219
+
220
+ portfolio_liabilities_metrics = self.convert_df_into_metrics(
221
+ dataloader.portfolio_liabilities_usd_df,
222
+ "portfolio_liabilities",
223
+ dataloader.portfolio.id,
224
+ val_date,
225
+ columns_map=dataloader.columns_map,
226
+ )
227
+ portfolio_aggregation_liabilities_metrics = self.convert_df_into_metrics(
228
+ dataloader.portfolio_liabilities_usd_df.aggregate,
229
+ "portfolio_liabilities",
230
+ dataloader.portfolio.id,
231
+ val_date,
232
+ columns_map=dataloader.columns_map,
233
+ dependency_metrics=portfolio_liabilities_metrics,
234
+ )
235
+
236
+ portfolio_aggregation_capital_metrics = self.convert_df_into_metrics(
237
+ dataloader.portfolio_aggregation_capital_employed,
238
+ "portfolio_capital_employed",
239
+ dataloader.portfolio.id,
240
+ val_date,
241
+ columns_map=dataloader.columns_map,
242
+ dependency_metrics=portfolio_aggregation_total_assets_metrics
243
+ + portfolio_aggregation_liabilities_metrics,
244
+ )
245
+
246
+ portfolio_aggregation_roce_metrics = self.convert_df_into_metrics(
247
+ dataloader.portfolio_aggregation_roce,
248
+ "portfolio_roce",
249
+ dataloader.portfolio.id,
250
+ val_date,
251
+ columns_map=dataloader.columns_map,
252
+ dependency_metrics=portfolio_aggregation_ebit_metrics + portfolio_aggregation_capital_metrics,
253
+ )
254
+
255
+ yield from portfolio_aggregation_roce_metrics