mirror of https://github.com/ghostfolio/ghostfolio
Browse Source
* Adding Coingecko Data Provider * Update changelog --------- Co-authored-by: Thomas <4159106+dtslvr@users.noreply.github.com>pull/1730/head
Mirio
2 years ago
committed by
GitHub
6 changed files with 200 additions and 3 deletions
@ -0,0 +1,191 @@ |
|||||
|
import { LookupItem } from '@ghostfolio/api/app/symbol/interfaces/lookup-item.interface'; |
||||
|
import { ConfigurationService } from '@ghostfolio/api/services/configuration.service'; |
||||
|
import { DataProviderInterface } from '@ghostfolio/api/services/data-provider/interfaces/data-provider.interface'; |
||||
|
import { |
||||
|
IDataProviderHistoricalResponse, |
||||
|
IDataProviderResponse |
||||
|
} from '@ghostfolio/api/services/interfaces/interfaces'; |
||||
|
import { DATE_FORMAT } from '@ghostfolio/common/helper'; |
||||
|
import { Granularity } from '@ghostfolio/common/types'; |
||||
|
import { Injectable, Logger } from '@nestjs/common'; |
||||
|
import { |
||||
|
AssetClass, |
||||
|
AssetSubClass, |
||||
|
DataSource, |
||||
|
SymbolProfile |
||||
|
} from '@prisma/client'; |
||||
|
import bent from 'bent'; |
||||
|
import { format, fromUnixTime, getUnixTime } from 'date-fns'; |
||||
|
|
||||
|
@Injectable() |
||||
|
export class CoinGeckoService implements DataProviderInterface { |
||||
|
private baseCurrency: string; |
||||
|
private readonly URL = 'https://api.coingecko.com/api/v3'; |
||||
|
|
||||
|
public constructor( |
||||
|
private readonly configurationService: ConfigurationService |
||||
|
) { |
||||
|
this.baseCurrency = this.configurationService.get('BASE_CURRENCY'); |
||||
|
} |
||||
|
|
||||
|
public canHandle(symbol: string) { |
||||
|
return true; |
||||
|
} |
||||
|
|
||||
|
public async getAssetProfile( |
||||
|
aSymbol: string |
||||
|
): Promise<Partial<SymbolProfile>> { |
||||
|
const response: Partial<SymbolProfile> = { |
||||
|
assetClass: AssetClass.CASH, |
||||
|
assetSubClass: AssetSubClass.CRYPTOCURRENCY, |
||||
|
currency: this.baseCurrency, |
||||
|
dataSource: this.getName(), |
||||
|
symbol: aSymbol |
||||
|
}; |
||||
|
|
||||
|
try { |
||||
|
const get = bent(`${this.URL}/coins/${aSymbol}`, 'GET', 'json', 200); |
||||
|
const { name } = await get(); |
||||
|
|
||||
|
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 get = bent( |
||||
|
`${ |
||||
|
this.URL |
||||
|
}/coins/${aSymbol}/market_chart/range?vs_currency=${this.baseCurrency.toLowerCase()}&from=${getUnixTime( |
||||
|
from |
||||
|
)}&to=${getUnixTime(to)}`,
|
||||
|
'GET', |
||||
|
'json', |
||||
|
200 |
||||
|
); |
||||
|
const { prices } = await get(); |
||||
|
|
||||
|
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( |
||||
|
aSymbols: string[] |
||||
|
): Promise<{ [symbol: string]: IDataProviderResponse }> { |
||||
|
const results: { [symbol: string]: IDataProviderResponse } = {}; |
||||
|
|
||||
|
if (aSymbols.length <= 0) { |
||||
|
return {}; |
||||
|
} |
||||
|
|
||||
|
try { |
||||
|
const get = bent( |
||||
|
`${this.URL}/simple/price?ids=${aSymbols.join( |
||||
|
',' |
||||
|
)}&vs_currencies=${this.baseCurrency.toLowerCase()}`,
|
||||
|
'GET', |
||||
|
'json', |
||||
|
200 |
||||
|
); |
||||
|
const response = await get(); |
||||
|
|
||||
|
for (const symbol in response) { |
||||
|
if (Object.prototype.hasOwnProperty.call(response, symbol)) { |
||||
|
results[symbol] = { |
||||
|
currency: this.baseCurrency, |
||||
|
dataSource: DataSource.COINGECKO, |
||||
|
marketPrice: response[symbol][this.baseCurrency.toLowerCase()], |
||||
|
marketState: 'open' |
||||
|
}; |
||||
|
} |
||||
|
} |
||||
|
} catch (error) { |
||||
|
Logger.error(error, 'CoinGeckoService'); |
||||
|
} |
||||
|
|
||||
|
return results; |
||||
|
} |
||||
|
|
||||
|
public async search(aQuery: string): Promise<{ items: LookupItem[] }> { |
||||
|
let items: LookupItem[] = []; |
||||
|
|
||||
|
if (aQuery.length <= 2) { |
||||
|
return { items }; |
||||
|
} |
||||
|
|
||||
|
try { |
||||
|
const get = bent( |
||||
|
`${this.URL}/search?query=${aQuery}`, |
||||
|
'GET', |
||||
|
'json', |
||||
|
200 |
||||
|
); |
||||
|
const { coins } = await get(); |
||||
|
|
||||
|
items = coins.map(({ id: symbol, name }) => { |
||||
|
return { |
||||
|
name, |
||||
|
symbol, |
||||
|
currency: this.baseCurrency, |
||||
|
dataSource: this.getName() |
||||
|
}; |
||||
|
}); |
||||
|
} catch (error) { |
||||
|
Logger.error(error, 'CoinGeckoService'); |
||||
|
} |
||||
|
|
||||
|
return { items }; |
||||
|
} |
||||
|
} |
@ -0,0 +1,2 @@ |
|||||
|
-- AlterEnum |
||||
|
ALTER TYPE "DataSource" ADD VALUE 'COINGECKO'; |
Loading…
Reference in new issue