|
@ -3,7 +3,6 @@ import { LookupItem } from '@ghostfolio/api/app/symbol/interfaces/lookup-item.in |
|
|
import { ConfigurationService } from '@ghostfolio/api/services/configuration/configuration.service'; |
|
|
import { ConfigurationService } from '@ghostfolio/api/services/configuration/configuration.service'; |
|
|
import { DataProviderInterface } from '@ghostfolio/api/services/data-provider/interfaces/data-provider.interface'; |
|
|
import { DataProviderInterface } from '@ghostfolio/api/services/data-provider/interfaces/data-provider.interface'; |
|
|
import { |
|
|
import { |
|
|
IDataGatheringItem, |
|
|
|
|
|
IDataProviderHistoricalResponse, |
|
|
IDataProviderHistoricalResponse, |
|
|
IDataProviderResponse |
|
|
IDataProviderResponse |
|
|
} from '@ghostfolio/api/services/interfaces/interfaces'; |
|
|
} from '@ghostfolio/api/services/interfaces/interfaces'; |
|
@ -12,6 +11,7 @@ import { PrismaService } from '@ghostfolio/api/services/prisma/prisma.service'; |
|
|
import { PropertyService } from '@ghostfolio/api/services/property/property.service'; |
|
|
import { PropertyService } from '@ghostfolio/api/services/property/property.service'; |
|
|
import { PROPERTY_DATA_SOURCE_MAPPING } from '@ghostfolio/common/config'; |
|
|
import { PROPERTY_DATA_SOURCE_MAPPING } from '@ghostfolio/common/config'; |
|
|
import { DATE_FORMAT, getStartOfUtcDate } from '@ghostfolio/common/helper'; |
|
|
import { DATE_FORMAT, getStartOfUtcDate } from '@ghostfolio/common/helper'; |
|
|
|
|
|
import { UniqueAsset } from '@ghostfolio/common/interfaces'; |
|
|
import type { Granularity, UserWithSettings } from '@ghostfolio/common/types'; |
|
|
import type { Granularity, UserWithSettings } from '@ghostfolio/common/types'; |
|
|
import { Inject, Injectable, Logger } from '@nestjs/common'; |
|
|
import { Inject, Injectable, Logger } from '@nestjs/common'; |
|
|
import { DataSource, MarketData, SymbolProfile } from '@prisma/client'; |
|
|
import { DataSource, MarketData, SymbolProfile } from '@prisma/client'; |
|
@ -45,12 +45,15 @@ export class DataProviderService { |
|
|
const dataProvider = this.getDataProvider(dataSource); |
|
|
const dataProvider = this.getDataProvider(dataSource); |
|
|
const symbol = dataProvider.getTestSymbol(); |
|
|
const symbol = dataProvider.getTestSymbol(); |
|
|
|
|
|
|
|
|
const quotes = await this.getQuotes([ |
|
|
const quotes = await this.getQuotes({ |
|
|
{ |
|
|
items: [ |
|
|
dataSource, |
|
|
{ |
|
|
symbol |
|
|
dataSource, |
|
|
} |
|
|
symbol |
|
|
]); |
|
|
} |
|
|
|
|
|
], |
|
|
|
|
|
useCache: false |
|
|
|
|
|
}); |
|
|
|
|
|
|
|
|
if (quotes[symbol]?.marketPrice > 0) { |
|
|
if (quotes[symbol]?.marketPrice > 0) { |
|
|
return true; |
|
|
return true; |
|
@ -59,14 +62,16 @@ export class DataProviderService { |
|
|
return false; |
|
|
return false; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
public async getAssetProfiles(items: IDataGatheringItem[]): Promise<{ |
|
|
public async getAssetProfiles(items: UniqueAsset[]): Promise<{ |
|
|
[symbol: string]: Partial<SymbolProfile>; |
|
|
[symbol: string]: Partial<SymbolProfile>; |
|
|
}> { |
|
|
}> { |
|
|
const response: { |
|
|
const response: { |
|
|
[symbol: string]: Partial<SymbolProfile>; |
|
|
[symbol: string]: Partial<SymbolProfile>; |
|
|
} = {}; |
|
|
} = {}; |
|
|
|
|
|
|
|
|
const itemsGroupedByDataSource = groupBy(items, (item) => item.dataSource); |
|
|
const itemsGroupedByDataSource = groupBy(items, ({ dataSource }) => { |
|
|
|
|
|
return dataSource; |
|
|
|
|
|
}); |
|
|
|
|
|
|
|
|
const promises = []; |
|
|
const promises = []; |
|
|
|
|
|
|
|
@ -127,7 +132,7 @@ export class DataProviderService { |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
public async getHistorical( |
|
|
public async getHistorical( |
|
|
aItems: IDataGatheringItem[], |
|
|
aItems: UniqueAsset[], |
|
|
aGranularity: Granularity = 'month', |
|
|
aGranularity: Granularity = 'month', |
|
|
from: Date, |
|
|
from: Date, |
|
|
to: Date |
|
|
to: Date |
|
@ -155,11 +160,11 @@ export class DataProviderService { |
|
|
)}'` |
|
|
)}'` |
|
|
: ''; |
|
|
: ''; |
|
|
|
|
|
|
|
|
const dataSources = aItems.map((item) => { |
|
|
const dataSources = aItems.map(({ dataSource }) => { |
|
|
return item.dataSource; |
|
|
return dataSource; |
|
|
}); |
|
|
}); |
|
|
const symbols = aItems.map((item) => { |
|
|
const symbols = aItems.map(({ symbol }) => { |
|
|
return item.symbol; |
|
|
return symbol; |
|
|
}); |
|
|
}); |
|
|
|
|
|
|
|
|
try { |
|
|
try { |
|
@ -192,7 +197,7 @@ export class DataProviderService { |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
public async getHistoricalRaw( |
|
|
public async getHistoricalRaw( |
|
|
aDataGatheringItems: IDataGatheringItem[], |
|
|
aDataGatheringItems: UniqueAsset[], |
|
|
from: Date, |
|
|
from: Date, |
|
|
to: Date |
|
|
to: Date |
|
|
): Promise<{ |
|
|
): Promise<{ |
|
@ -229,7 +234,13 @@ export class DataProviderService { |
|
|
return result; |
|
|
return result; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
public async getQuotes(items: IDataGatheringItem[]): Promise<{ |
|
|
public async getQuotes({ |
|
|
|
|
|
items, |
|
|
|
|
|
useCache = true |
|
|
|
|
|
}: { |
|
|
|
|
|
items: UniqueAsset[]; |
|
|
|
|
|
useCache?: boolean; |
|
|
|
|
|
}): Promise<{ |
|
|
[symbol: string]: IDataProviderResponse; |
|
|
[symbol: string]: IDataProviderResponse; |
|
|
}> { |
|
|
}> { |
|
|
const response: { |
|
|
const response: { |
|
@ -238,23 +249,24 @@ export class DataProviderService { |
|
|
const startTimeTotal = performance.now(); |
|
|
const startTimeTotal = performance.now(); |
|
|
|
|
|
|
|
|
// Get items from cache
|
|
|
// Get items from cache
|
|
|
const itemsToFetch: IDataGatheringItem[] = []; |
|
|
const itemsToFetch: UniqueAsset[] = []; |
|
|
|
|
|
|
|
|
for (const { dataSource, symbol } of items) { |
|
|
for (const { dataSource, symbol } of items) { |
|
|
const quoteString = await this.redisCacheService.get( |
|
|
if (useCache) { |
|
|
this.redisCacheService.getQuoteKey({ dataSource, symbol }) |
|
|
const quoteString = await this.redisCacheService.get( |
|
|
); |
|
|
this.redisCacheService.getQuoteKey({ dataSource, symbol }) |
|
|
|
|
|
); |
|
|
|
|
|
|
|
|
if (quoteString) { |
|
|
if (quoteString) { |
|
|
try { |
|
|
try { |
|
|
const cachedDataProviderResponse = JSON.parse(quoteString); |
|
|
const cachedDataProviderResponse = JSON.parse(quoteString); |
|
|
response[symbol] = cachedDataProviderResponse; |
|
|
response[symbol] = cachedDataProviderResponse; |
|
|
} catch {} |
|
|
continue; |
|
|
|
|
|
} catch {} |
|
|
|
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
if (!quoteString) { |
|
|
itemsToFetch.push({ dataSource, symbol }); |
|
|
itemsToFetch.push({ dataSource, symbol }); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
const numberOfItemsInCache = Object.keys(response)?.length; |
|
|
const numberOfItemsInCache = Object.keys(response)?.length; |
|
|