import { LookupItem } from '@ghostfolio/api/app/symbol/interfaces/lookup-item.interface'; import { DataProviderInterface } from '@ghostfolio/api/services/data-provider/interfaces/data-provider.interface'; import { IDataProviderHistoricalResponse, IDataProviderResponse } from '@ghostfolio/api/services/interfaces/interfaces'; import { DEFAULT_CURRENCY, DEFAULT_REQUEST_TIMEOUT } from '@ghostfolio/common/config'; import { DATE_FORMAT } from '@ghostfolio/common/helper'; import { DataProviderInfo } from '@ghostfolio/common/interfaces'; import { Granularity } from '@ghostfolio/common/types'; import { Injectable, Logger } from '@nestjs/common'; import { AssetClass, AssetSubClass, DataSource, SymbolProfile } from '@prisma/client'; import { format, fromUnixTime, getUnixTime } from 'date-fns'; import got from 'got'; @Injectable() export class CoinGeckoService implements DataProviderInterface { private readonly URL = 'https://api.coingecko.com/api/v3'; public constructor() {} public canHandle(symbol: string) { return true; } public async getAssetProfile( aSymbol: string ): Promise> { const response: Partial = { assetClass: AssetClass.CASH, assetSubClass: AssetSubClass.CRYPTOCURRENCY, currency: DEFAULT_CURRENCY, dataSource: this.getName(), symbol: aSymbol }; try { const abortController = new AbortController(); setTimeout(() => { abortController.abort(); }, DEFAULT_REQUEST_TIMEOUT); const { name } = await got(`${this.URL}/coins/${aSymbol}`, { // @ts-ignore signal: abortController.signal }).json(); response.name = name; } catch (error) { Logger.error(error, 'CoinGeckoService'); } return response; } public async getDividends({ from, granularity = 'day', symbol, to }: { from: Date; granularity: Granularity; symbol: string; to: Date; }) { return {}; } public async getHistorical( aSymbol: string, aGranularity: Granularity = 'day', from: Date, to: Date ): Promise<{ [symbol: string]: { [date: string]: IDataProviderHistoricalResponse }; }> { try { const abortController = new AbortController(); setTimeout(() => { abortController.abort(); }, DEFAULT_REQUEST_TIMEOUT); const { prices } = await got( `${ this.URL }/coins/${aSymbol}/market_chart/range?vs_currency=${DEFAULT_CURRENCY.toLowerCase()}&from=${getUnixTime( from )}&to=${getUnixTime(to)}`, { // @ts-ignore signal: abortController.signal } ).json(); const result: { [symbol: string]: { [date: string]: IDataProviderHistoricalResponse }; } = { [aSymbol]: {} }; for (const [timestamp, marketPrice] of prices) { result[aSymbol][format(fromUnixTime(timestamp / 1000), DATE_FORMAT)] = { marketPrice }; } return result; } catch (error) { throw new Error( `Could not get historical market data for ${aSymbol} (${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({ symbols }: { symbols: string[]; }): Promise<{ [symbol: string]: IDataProviderResponse }> { const response: { [symbol: string]: IDataProviderResponse } = {}; if (symbols.length <= 0) { return response; } try { const abortController = new AbortController(); setTimeout(() => { abortController.abort(); }, DEFAULT_REQUEST_TIMEOUT); const response = await got( `${this.URL}/simple/price?ids=${symbols.join( ',' )}&vs_currencies=${DEFAULT_CURRENCY.toLowerCase()}`, { // @ts-ignore signal: abortController.signal } ).json(); for (const symbol in response) { if (Object.prototype.hasOwnProperty.call(response, symbol)) { response[symbol] = { currency: DEFAULT_CURRENCY, dataProviderInfo: this.getDataProviderInfo(), dataSource: DataSource.COINGECKO, marketPrice: response[symbol][DEFAULT_CURRENCY.toLowerCase()], marketState: 'open' }; } } } catch (error) { Logger.error(error, 'CoinGeckoService'); } return response; } public getTestSymbol() { return 'bitcoin'; } public async search({ includeIndices = false, query }: { includeIndices?: boolean; query: string; }): Promise<{ items: LookupItem[] }> { let items: LookupItem[] = []; try { const abortController = new AbortController(); setTimeout(() => { abortController.abort(); }, DEFAULT_REQUEST_TIMEOUT); const { coins } = await got(`${this.URL}/search?query=${query}`, { // @ts-ignore signal: abortController.signal }).json(); items = coins.map(({ id: symbol, name }) => { return { name, symbol, assetClass: AssetClass.CASH, assetSubClass: AssetSubClass.CRYPTOCURRENCY, currency: DEFAULT_CURRENCY, dataSource: this.getName() }; }); } catch (error) { Logger.error(error, 'CoinGeckoService'); } return { items }; } private getDataProviderInfo(): DataProviderInfo { return { name: 'CoinGecko', url: 'https://coingecko.com' }; } }