import { ConfigurationService } from '@ghostfolio/api/services/configuration/configuration.service'; import { DataProviderInterface, GetAssetProfileParams, GetDividendsParams, GetHistoricalParams, GetQuotesParams, GetSearchParams } from '@ghostfolio/api/services/data-provider/interfaces/data-provider.interface'; import { IDataProviderHistoricalResponse, IDataProviderResponse } from '@ghostfolio/api/services/interfaces/interfaces'; import { DEFAULT_CURRENCY } from '@ghostfolio/common/config'; import { DATE_FORMAT } from '@ghostfolio/common/helper'; import { DataProviderInfo, LookupItem, LookupResponse } from '@ghostfolio/common/interfaces'; import { Injectable, Logger } from '@nestjs/common'; import { AssetClass, AssetSubClass, DataSource, SymbolProfile } from '@prisma/client'; import { format, fromUnixTime, getUnixTime } from 'date-fns'; @Injectable() export class CoinGeckoService implements DataProviderInterface { private readonly apiUrl: string; private readonly headers: HeadersInit = {}; public constructor( private readonly configurationService: ConfigurationService ) { const apiKeyDemo = this.configurationService.get('API_KEY_COINGECKO_DEMO'); const apiKeyPro = this.configurationService.get('API_KEY_COINGECKO_PRO'); this.apiUrl = 'https://api.coingecko.com/api/v3'; if (apiKeyDemo) { this.headers['x-cg-demo-api-key'] = apiKeyDemo; } if (apiKeyPro) { this.apiUrl = 'https://pro-api.coingecko.com/api/v3'; this.headers['x-cg-pro-api-key'] = apiKeyPro; } } public canHandle() { return true; } public async getAssetProfile({ symbol }: GetAssetProfileParams): Promise> { const response: Partial = { symbol, assetClass: AssetClass.LIQUIDITY, assetSubClass: AssetSubClass.CRYPTOCURRENCY, currency: DEFAULT_CURRENCY, dataSource: this.getName() }; try { const { name } = await fetch(`${this.apiUrl}/coins/${symbol}`, { headers: this.headers, signal: AbortSignal.timeout( this.configurationService.get('REQUEST_TIMEOUT') ) }).then((res) => res.json()); response.name = name; } catch (error) { let message = error; if (['AbortError', 'TimeoutError'].includes(error?.name)) { message = `RequestError: The operation to get the asset profile for ${symbol} was aborted because the request to the data provider took more than ${( this.configurationService.get('REQUEST_TIMEOUT') / 1000 ).toFixed(3)} seconds`; } Logger.error(message, 'CoinGeckoService'); } return response; } public getDataProviderInfo(): DataProviderInfo { return { dataSource: DataSource.COINGECKO, isPremium: false, name: 'CoinGecko', url: 'https://coingecko.com' }; } public async getDividends({}: GetDividendsParams) { return {}; } public async getHistorical({ from, requestTimeout = this.configurationService.get('REQUEST_TIMEOUT'), symbol, to }: GetHistoricalParams): Promise<{ [symbol: string]: { [date: string]: IDataProviderHistoricalResponse }; }> { try { const { error, prices, status } = await fetch( `${ this.apiUrl }/coins/${symbol}/market_chart/range?vs_currency=${DEFAULT_CURRENCY.toLowerCase()}&from=${getUnixTime( from )}&to=${getUnixTime(to)}`, { headers: this.headers, signal: AbortSignal.timeout(requestTimeout) } ).then((res) => res.json()); if (error?.status) { throw new Error(error.status.error_message); } if (status) { throw new Error(status.error_message); } const result: { [symbol: string]: { [date: string]: IDataProviderHistoricalResponse }; } = { [symbol]: {} }; for (const [timestamp, marketPrice] of prices) { result[symbol][format(fromUnixTime(timestamp / 1000), DATE_FORMAT)] = { marketPrice }; } return result; } catch (error) { throw new Error( `Could not get historical market data for ${symbol} (${this.getName()}) from ${format( from, DATE_FORMAT )} to ${format(to, DATE_FORMAT)}: [${error.name}] ${error.message}` ); } } public getMaxNumberOfSymbolsPerRequest() { return 50; } public getName(): DataSource { return DataSource.COINGECKO; } public async getQuotes({ requestTimeout = this.configurationService.get('REQUEST_TIMEOUT'), symbols }: GetQuotesParams): Promise<{ [symbol: string]: IDataProviderResponse }> { const response: { [symbol: string]: IDataProviderResponse } = {}; if (symbols.length <= 0) { return response; } try { const quotes = await fetch( `${this.apiUrl}/simple/price?ids=${symbols.join( ',' )}&vs_currencies=${DEFAULT_CURRENCY.toLowerCase()}`, { headers: this.headers, signal: AbortSignal.timeout(requestTimeout) } ).then((res) => res.json()); for (const symbol in quotes) { response[symbol] = { currency: DEFAULT_CURRENCY, dataProviderInfo: this.getDataProviderInfo(), dataSource: DataSource.COINGECKO, marketPrice: quotes[symbol][DEFAULT_CURRENCY.toLowerCase()], marketState: 'open' }; } } catch (error) { let message = error; if (['AbortError', 'TimeoutError'].includes(error?.name)) { message = `RequestError: The operation to get the quotes for ${symbols.join( ', ' )} was aborted because the request to the data provider took more than ${( this.configurationService.get('REQUEST_TIMEOUT') / 1000 ).toFixed(3)} seconds`; } Logger.error(message, 'CoinGeckoService'); } return response; } public getTestSymbol() { return 'bitcoin'; } public async search({ query }: GetSearchParams): Promise { let items: LookupItem[] = []; try { const { coins } = await fetch(`${this.apiUrl}/search?query=${query}`, { headers: this.headers, signal: AbortSignal.timeout( this.configurationService.get('REQUEST_TIMEOUT') ) }).then((res) => res.json()); items = coins.map(({ id: symbol, name }) => { return { name, symbol, assetClass: AssetClass.LIQUIDITY, assetSubClass: AssetSubClass.CRYPTOCURRENCY, currency: DEFAULT_CURRENCY, dataProviderInfo: this.getDataProviderInfo(), dataSource: this.getName() }; }); } catch (error) { let message = error; if (['AbortError', 'TimeoutError'].includes(error?.name)) { message = `RequestError: The operation to search for ${query} was aborted because the request to the data provider took more than ${( this.configurationService.get('REQUEST_TIMEOUT') / 1000 ).toFixed(3)} seconds`; } Logger.error(message, 'CoinGeckoService'); } return { items }; } }