Browse Source

Feature/add and update historical data in bulk (#1904)

* Upsert historical data in bulk

* Update changelog
pull/1905/head
Thomas Kaul 2 years ago
committed by GitHub
parent
commit
623a284ba4
No known key found for this signature in database GPG Key ID: 4AEE18F83AFDEB23
  1. 4
      CHANGELOG.md
  2. 48
      apps/api/src/services/data-gathering/data-gathering.processor.ts
  3. 38
      test/import/ok-vti-buy-long-history.json

4
CHANGELOG.md

@ -11,6 +11,10 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- Extended the support of the impersonation mode for local development - Extended the support of the impersonation mode for local development
### Changed
- Improved the queue jobs implementation by adding / updating historical market data in bulk
### Fixed ### Fixed
- Improved the holdings table by showing the cash position also when the filter contains the accounts, so that we can see the total allocation for that account - Improved the holdings table by showing the cash position also when the filter contains the accounts, so that we can see the total allocation for that account

48
apps/api/src/services/data-gathering/data-gathering.processor.ts

@ -1,6 +1,5 @@
import { DataProviderService } from '@ghostfolio/api/services/data-provider/data-provider.service'; import { DataProviderService } from '@ghostfolio/api/services/data-provider/data-provider.service';
import { IDataGatheringItem } from '@ghostfolio/api/services/interfaces/interfaces'; import { IDataGatheringItem } from '@ghostfolio/api/services/interfaces/interfaces';
import { PrismaService } from '@ghostfolio/api/services/prisma/prisma.service';
import { import {
DATA_GATHERING_QUEUE, DATA_GATHERING_QUEUE,
GATHER_ASSET_PROFILE_PROCESS, GATHER_ASSET_PROFILE_PROCESS,
@ -11,6 +10,7 @@ import { UniqueAsset } from '@ghostfolio/common/interfaces';
import { MarketDataService } from '@ghostfolio/api/services/market-data/market-data.service'; import { MarketDataService } from '@ghostfolio/api/services/market-data/market-data.service';
import { Process, Processor } from '@nestjs/bull'; import { Process, Processor } from '@nestjs/bull';
import { Injectable, Logger } from '@nestjs/common'; import { Injectable, Logger } from '@nestjs/common';
import { Prisma } from '@prisma/client';
import { Job } from 'bull'; import { Job } from 'bull';
import { import {
format, format,
@ -32,7 +32,7 @@ export class DataGatheringProcessor {
private readonly marketDataService: MarketDataService private readonly marketDataService: MarketDataService
) {} ) {}
@Process(GATHER_ASSET_PROFILE_PROCESS) @Process({ concurrency: 1, name: GATHER_ASSET_PROFILE_PROCESS })
public async gatherAssetProfile(job: Job<UniqueAsset>) { public async gatherAssetProfile(job: Job<UniqueAsset>) {
try { try {
await this.dataGatheringService.gatherAssetProfiles([job.data]); await this.dataGatheringService.gatherAssetProfiles([job.data]);
@ -46,18 +46,27 @@ export class DataGatheringProcessor {
} }
} }
@Process(GATHER_HISTORICAL_MARKET_DATA_PROCESS) @Process({ concurrency: 1, name: GATHER_HISTORICAL_MARKET_DATA_PROCESS })
public async gatherHistoricalMarketData(job: Job<IDataGatheringItem>) { public async gatherHistoricalMarketData(job: Job<IDataGatheringItem>) {
try { try {
const { dataSource, date, symbol } = job.data; const { dataSource, date, symbol } = job.data;
let currentDate = parseISO(<string>(<unknown>date));
Logger.log(
`Historical market data gathering has been started for ${symbol} (${dataSource}) at ${format(
currentDate,
DATE_FORMAT
)}`,
`DataGatheringProcessor (${GATHER_HISTORICAL_MARKET_DATA_PROCESS})`
);
const historicalData = await this.dataProviderService.getHistoricalRaw( const historicalData = await this.dataProviderService.getHistoricalRaw(
[{ dataSource, symbol }], [{ dataSource, symbol }],
parseISO(<string>(<unknown>date)), currentDate,
new Date() new Date()
); );
let currentDate = parseISO(<string>(<unknown>date)); const data: Prisma.MarketDataUpdateInput[] = [];
let lastMarketPrice: number; let lastMarketPrice: number;
while ( while (
@ -83,21 +92,13 @@ export class DataGatheringProcessor {
} }
if (lastMarketPrice) { if (lastMarketPrice) {
try { data.push({
await this.marketDataService.updateMarketData({ dataSource,
data: { symbol,
marketPrice: lastMarketPrice, date: getStartOfUtcDate(currentDate),
state: 'CLOSE' marketPrice: lastMarketPrice,
}, state: 'CLOSE'
where: { });
dataSource_date_symbol: {
dataSource,
symbol,
date: getStartOfUtcDate(currentDate)
}
}
});
} catch {}
} }
// Count month one up for iteration // Count month one up for iteration
@ -111,8 +112,13 @@ export class DataGatheringProcessor {
); );
} }
await this.marketDataService.updateMany({ data });
Logger.log( Logger.log(
`Historical market data gathering has been completed for ${symbol} (${dataSource}).`, `Historical market data gathering has been completed for ${symbol} (${dataSource}) at ${format(
currentDate,
DATE_FORMAT
)}`,
`DataGatheringProcessor (${GATHER_HISTORICAL_MARKET_DATA_PROCESS})` `DataGatheringProcessor (${GATHER_HISTORICAL_MARKET_DATA_PROCESS})`
); );
} catch (error) { } catch (error) {

38
test/import/ok-vti-buy-long-history.json

@ -0,0 +1,38 @@
{
"meta": {
"date": "2023-04-29T00:00:00.000Z",
"version": "dev"
},
"activities": [
{
"fee": 0,
"quantity": 10,
"type": "BUY",
"unitPrice": 65.31,
"currency": "USD",
"dataSource": "YAHOO",
"date": "2012-01-02T22:00:00.000Z",
"symbol": "VTI"
},
{
"fee": 0,
"quantity": 10,
"type": "BUY",
"unitPrice": 65.40,
"currency": "USD",
"dataSource": "YAHOO",
"date": "2011-01-02T22:00:00.000Z",
"symbol": "VTI"
},
{
"fee": 0,
"quantity": 10,
"type": "BUY",
"unitPrice": 57.05,
"currency": "USD",
"dataSource": "YAHOO",
"date": "2010-01-03T22:00:00.000Z",
"symbol": "VTI"
}
]
}
Loading…
Cancel
Save