|
|
@ -50,6 +50,7 @@ import { |
|
|
|
PortfolioPerformanceResponse, |
|
|
|
PortfolioPosition, |
|
|
|
PortfolioReportResponse, |
|
|
|
PortfolioReportRule, |
|
|
|
PortfolioSummary, |
|
|
|
UserSettings |
|
|
|
} from '@ghostfolio/common/interfaces'; |
|
|
@ -1231,48 +1232,54 @@ export class PortfolioService { |
|
|
|
}) |
|
|
|
).toNumber(); |
|
|
|
|
|
|
|
const rules: PortfolioReportResponse['xRay']['rules'] = { |
|
|
|
accountClusterRisk: |
|
|
|
summary.activityCount > 0 |
|
|
|
? await this.rulesService.evaluate( |
|
|
|
const categories: PortfolioReportResponse['xRay']['categories'] = [ |
|
|
|
{ |
|
|
|
key: 'liquidity', |
|
|
|
name: this.i18nService.getTranslation({ |
|
|
|
id: 'rule.liquidity.category', |
|
|
|
languageCode: userSettings.language |
|
|
|
}), |
|
|
|
rules: await this.rulesService.evaluate( |
|
|
|
[ |
|
|
|
new AccountClusterRiskCurrentInvestment( |
|
|
|
this.exchangeRateDataService, |
|
|
|
this.i18nService, |
|
|
|
userSettings.language, |
|
|
|
accounts |
|
|
|
), |
|
|
|
new AccountClusterRiskSingleAccount( |
|
|
|
new BuyingPower( |
|
|
|
this.exchangeRateDataService, |
|
|
|
this.i18nService, |
|
|
|
userSettings.language, |
|
|
|
accounts |
|
|
|
summary.cash, |
|
|
|
userSettings.language |
|
|
|
) |
|
|
|
], |
|
|
|
userSettings |
|
|
|
) |
|
|
|
: undefined, |
|
|
|
assetClassClusterRisk: |
|
|
|
summary.activityCount > 0 |
|
|
|
? await this.rulesService.evaluate( |
|
|
|
}, |
|
|
|
{ |
|
|
|
key: 'emergencyFund', |
|
|
|
name: this.i18nService.getTranslation({ |
|
|
|
id: 'rule.emergencyFund.category', |
|
|
|
languageCode: userSettings.language |
|
|
|
}), |
|
|
|
rules: await this.rulesService.evaluate( |
|
|
|
[ |
|
|
|
new AssetClassClusterRiskEquity( |
|
|
|
this.exchangeRateDataService, |
|
|
|
this.i18nService, |
|
|
|
userSettings.language, |
|
|
|
Object.values(holdings) |
|
|
|
), |
|
|
|
new AssetClassClusterRiskFixedIncome( |
|
|
|
new EmergencyFundSetup( |
|
|
|
this.exchangeRateDataService, |
|
|
|
this.i18nService, |
|
|
|
userSettings.language, |
|
|
|
Object.values(holdings) |
|
|
|
this.getTotalEmergencyFund({ |
|
|
|
userSettings, |
|
|
|
emergencyFundHoldingsValueInBaseCurrency: |
|
|
|
this.getEmergencyFundHoldingsValueInBaseCurrency({ holdings }) |
|
|
|
}).toNumber() |
|
|
|
) |
|
|
|
], |
|
|
|
userSettings |
|
|
|
) |
|
|
|
: undefined, |
|
|
|
currencyClusterRisk: |
|
|
|
}, |
|
|
|
{ |
|
|
|
key: 'currencyClusterRisk', |
|
|
|
name: this.i18nService.getTranslation({ |
|
|
|
id: 'rule.currencyClusterRisk.category', |
|
|
|
languageCode: userSettings.language |
|
|
|
}), |
|
|
|
rules: |
|
|
|
summary.activityCount > 0 |
|
|
|
? await this.rulesService.evaluate( |
|
|
|
[ |
|
|
@ -1291,68 +1298,98 @@ export class PortfolioService { |
|
|
|
], |
|
|
|
userSettings |
|
|
|
) |
|
|
|
: undefined, |
|
|
|
economicMarketClusterRisk: |
|
|
|
: undefined |
|
|
|
}, |
|
|
|
{ |
|
|
|
key: 'assetClassClusterRisk', |
|
|
|
name: this.i18nService.getTranslation({ |
|
|
|
id: 'rule.assetClassClusterRisk.category', |
|
|
|
languageCode: userSettings.language |
|
|
|
}), |
|
|
|
rules: |
|
|
|
summary.activityCount > 0 |
|
|
|
? await this.rulesService.evaluate( |
|
|
|
[ |
|
|
|
new EconomicMarketClusterRiskDevelopedMarkets( |
|
|
|
new AssetClassClusterRiskEquity( |
|
|
|
this.exchangeRateDataService, |
|
|
|
this.i18nService, |
|
|
|
marketsTotalInBaseCurrency, |
|
|
|
markets.developedMarkets.valueInBaseCurrency, |
|
|
|
userSettings.language |
|
|
|
userSettings.language, |
|
|
|
Object.values(holdings) |
|
|
|
), |
|
|
|
new EconomicMarketClusterRiskEmergingMarkets( |
|
|
|
new AssetClassClusterRiskFixedIncome( |
|
|
|
this.exchangeRateDataService, |
|
|
|
this.i18nService, |
|
|
|
marketsTotalInBaseCurrency, |
|
|
|
markets.emergingMarkets.valueInBaseCurrency, |
|
|
|
userSettings.language |
|
|
|
userSettings.language, |
|
|
|
Object.values(holdings) |
|
|
|
) |
|
|
|
], |
|
|
|
userSettings |
|
|
|
) |
|
|
|
: undefined, |
|
|
|
emergencyFund: await this.rulesService.evaluate( |
|
|
|
: undefined |
|
|
|
}, |
|
|
|
{ |
|
|
|
key: 'accountClusterRisk', |
|
|
|
name: this.i18nService.getTranslation({ |
|
|
|
id: 'rule.accountClusterRisk.category', |
|
|
|
languageCode: userSettings.language |
|
|
|
}), |
|
|
|
rules: |
|
|
|
summary.activityCount > 0 |
|
|
|
? await this.rulesService.evaluate( |
|
|
|
[ |
|
|
|
new EmergencyFundSetup( |
|
|
|
new AccountClusterRiskCurrentInvestment( |
|
|
|
this.exchangeRateDataService, |
|
|
|
this.i18nService, |
|
|
|
userSettings.language, |
|
|
|
this.getTotalEmergencyFund({ |
|
|
|
userSettings, |
|
|
|
emergencyFundHoldingsValueInBaseCurrency: |
|
|
|
this.getEmergencyFundHoldingsValueInBaseCurrency({ holdings }) |
|
|
|
}).toNumber() |
|
|
|
) |
|
|
|
], |
|
|
|
userSettings |
|
|
|
accounts |
|
|
|
), |
|
|
|
fees: await this.rulesService.evaluate( |
|
|
|
[ |
|
|
|
new FeeRatioInitialInvestment( |
|
|
|
new AccountClusterRiskSingleAccount( |
|
|
|
this.exchangeRateDataService, |
|
|
|
this.i18nService, |
|
|
|
userSettings.language, |
|
|
|
summary.committedFunds, |
|
|
|
summary.fees |
|
|
|
accounts |
|
|
|
) |
|
|
|
], |
|
|
|
userSettings |
|
|
|
), |
|
|
|
liquidity: await this.rulesService.evaluate( |
|
|
|
) |
|
|
|
: undefined |
|
|
|
}, |
|
|
|
{ |
|
|
|
key: 'economicMarketClusterRisk', |
|
|
|
name: this.i18nService.getTranslation({ |
|
|
|
id: 'rule.economicMarketClusterRisk.category', |
|
|
|
languageCode: userSettings.language |
|
|
|
}), |
|
|
|
rules: |
|
|
|
summary.activityCount > 0 |
|
|
|
? await this.rulesService.evaluate( |
|
|
|
[ |
|
|
|
new BuyingPower( |
|
|
|
new EconomicMarketClusterRiskDevelopedMarkets( |
|
|
|
this.exchangeRateDataService, |
|
|
|
this.i18nService, |
|
|
|
summary.cash, |
|
|
|
marketsTotalInBaseCurrency, |
|
|
|
markets.developedMarkets.valueInBaseCurrency, |
|
|
|
userSettings.language |
|
|
|
), |
|
|
|
new EconomicMarketClusterRiskEmergingMarkets( |
|
|
|
this.exchangeRateDataService, |
|
|
|
this.i18nService, |
|
|
|
marketsTotalInBaseCurrency, |
|
|
|
markets.emergingMarkets.valueInBaseCurrency, |
|
|
|
userSettings.language |
|
|
|
) |
|
|
|
], |
|
|
|
userSettings |
|
|
|
), |
|
|
|
regionalMarketClusterRisk: |
|
|
|
) |
|
|
|
: undefined |
|
|
|
}, |
|
|
|
{ |
|
|
|
key: 'regionalMarketClusterRisk', |
|
|
|
name: this.i18nService.getTranslation({ |
|
|
|
id: 'rule.regionalMarketClusterRisk.category', |
|
|
|
languageCode: userSettings.language |
|
|
|
}), |
|
|
|
rules: |
|
|
|
summary.activityCount > 0 |
|
|
|
? await this.rulesService.evaluate( |
|
|
|
[ |
|
|
@ -1395,12 +1432,36 @@ export class PortfolioService { |
|
|
|
userSettings |
|
|
|
) |
|
|
|
: undefined |
|
|
|
}; |
|
|
|
}, |
|
|
|
{ |
|
|
|
key: 'fees', |
|
|
|
name: this.i18nService.getTranslation({ |
|
|
|
id: 'rule.fees.category', |
|
|
|
languageCode: userSettings.language |
|
|
|
}), |
|
|
|
rules: await this.rulesService.evaluate( |
|
|
|
[ |
|
|
|
new FeeRatioInitialInvestment( |
|
|
|
this.exchangeRateDataService, |
|
|
|
this.i18nService, |
|
|
|
userSettings.language, |
|
|
|
summary.committedFunds, |
|
|
|
summary.fees |
|
|
|
) |
|
|
|
], |
|
|
|
userSettings |
|
|
|
) |
|
|
|
} |
|
|
|
]; |
|
|
|
|
|
|
|
return { |
|
|
|
xRay: { |
|
|
|
rules, |
|
|
|
statistics: this.getReportStatistics(rules) |
|
|
|
categories, |
|
|
|
statistics: this.getReportStatistics( |
|
|
|
categories.flatMap(({ rules }) => { |
|
|
|
return rules ?? []; |
|
|
|
}) |
|
|
|
) |
|
|
|
} |
|
|
|
}; |
|
|
|
} |
|
|
@ -1822,7 +1883,7 @@ export class PortfolioService { |
|
|
|
} |
|
|
|
|
|
|
|
private getReportStatistics( |
|
|
|
evaluatedRules: PortfolioReportResponse['xRay']['rules'] |
|
|
|
evaluatedRules: PortfolioReportRule[] |
|
|
|
): PortfolioReportResponse['xRay']['statistics'] { |
|
|
|
const rulesActiveCount = Object.values(evaluatedRules) |
|
|
|
.flat() |
|
|
|