|
@ -43,16 +43,12 @@ export class YahooFinanceService implements DataProviderInterface { |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
public async get( |
|
|
public async get( |
|
|
aSymbols: string[] |
|
|
aYahooFinanceSymbols: string[] |
|
|
): Promise<{ [symbol: string]: IDataProviderResponse }> { |
|
|
): Promise<{ [symbol: string]: IDataProviderResponse }> { |
|
|
if (aSymbols.length <= 0) { |
|
|
if (aYahooFinanceSymbols.length <= 0) { |
|
|
return {}; |
|
|
return {}; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
const yahooSymbols = aSymbols.map((symbol) => { |
|
|
|
|
|
return this.convertToYahooSymbol(symbol); |
|
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
|
|
try { |
|
|
try { |
|
|
const response: { [symbol: string]: IDataProviderResponse } = {}; |
|
|
const response: { [symbol: string]: IDataProviderResponse } = {}; |
|
|
|
|
|
|
|
@ -60,12 +56,12 @@ export class YahooFinanceService implements DataProviderInterface { |
|
|
[symbol: string]: IYahooFinanceQuoteResponse; |
|
|
[symbol: string]: IYahooFinanceQuoteResponse; |
|
|
} = await yahooFinance.quote({ |
|
|
} = await yahooFinance.quote({ |
|
|
modules: ['price', 'summaryProfile'], |
|
|
modules: ['price', 'summaryProfile'], |
|
|
symbols: yahooSymbols |
|
|
symbols: aYahooFinanceSymbols |
|
|
}); |
|
|
}); |
|
|
|
|
|
|
|
|
for (const [yahooSymbol, value] of Object.entries(data)) { |
|
|
for (const [yahooFinanceSymbol, value] of Object.entries(data)) { |
|
|
// Convert symbols back
|
|
|
// Convert symbols back
|
|
|
const symbol = convertFromYahooSymbol(yahooSymbol); |
|
|
const symbol = convertFromYahooFinanceSymbol(yahooFinanceSymbol); |
|
|
|
|
|
|
|
|
const { assetClass, assetSubClass } = this.parseAssetClass(value.price); |
|
|
const { assetClass, assetSubClass } = this.parseAssetClass(value.price); |
|
|
|
|
|
|
|
@ -136,15 +132,15 @@ export class YahooFinanceService implements DataProviderInterface { |
|
|
return {}; |
|
|
return {}; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
const yahooSymbols = aSymbols.map((symbol) => { |
|
|
const yahooFinanceSymbols = aSymbols.map((symbol) => { |
|
|
return this.convertToYahooSymbol(symbol); |
|
|
return convertToYahooFinanceSymbol(symbol); |
|
|
}); |
|
|
}); |
|
|
|
|
|
|
|
|
try { |
|
|
try { |
|
|
const historicalData: { |
|
|
const historicalData: { |
|
|
[symbol: string]: IYahooFinanceHistoricalResponse[]; |
|
|
[symbol: string]: IYahooFinanceHistoricalResponse[]; |
|
|
} = await yahooFinance.historical({ |
|
|
} = await yahooFinance.historical({ |
|
|
symbols: yahooSymbols, |
|
|
symbols: yahooFinanceSymbols, |
|
|
from: format(from, DATE_FORMAT), |
|
|
from: format(from, DATE_FORMAT), |
|
|
to: format(to, DATE_FORMAT) |
|
|
to: format(to, DATE_FORMAT) |
|
|
}); |
|
|
}); |
|
@ -153,9 +149,11 @@ export class YahooFinanceService implements DataProviderInterface { |
|
|
[symbol: string]: { [date: string]: IDataProviderHistoricalResponse }; |
|
|
[symbol: string]: { [date: string]: IDataProviderHistoricalResponse }; |
|
|
} = {}; |
|
|
} = {}; |
|
|
|
|
|
|
|
|
for (const [yahooSymbol, timeSeries] of Object.entries(historicalData)) { |
|
|
for (const [yahooFinanceSymbol, timeSeries] of Object.entries( |
|
|
|
|
|
historicalData |
|
|
|
|
|
)) { |
|
|
// Convert symbols back
|
|
|
// Convert symbols back
|
|
|
const symbol = convertFromYahooSymbol(yahooSymbol); |
|
|
const symbol = convertFromYahooFinanceSymbol(yahooFinanceSymbol); |
|
|
response[symbol] = {}; |
|
|
response[symbol] = {}; |
|
|
|
|
|
|
|
|
timeSeries.forEach((timeSerie) => { |
|
|
timeSeries.forEach((timeSerie) => { |
|
@ -175,7 +173,7 @@ export class YahooFinanceService implements DataProviderInterface { |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
public async search(aSymbol: string): Promise<{ items: LookupItem[] }> { |
|
|
public async search(aSymbol: string): Promise<{ items: LookupItem[] }> { |
|
|
let items: LookupItem[] = []; |
|
|
const items: LookupItem[] = []; |
|
|
|
|
|
|
|
|
try { |
|
|
try { |
|
|
const get = bent( |
|
|
const get = bent( |
|
@ -192,19 +190,6 @@ export class YahooFinanceService implements DataProviderInterface { |
|
|
// filter out undefined symbols
|
|
|
// filter out undefined symbols
|
|
|
return quote.symbol; |
|
|
return quote.symbol; |
|
|
}) |
|
|
}) |
|
|
.filter(({ quoteType }) => { |
|
|
|
|
|
return quoteType === 'EQUITY' || quoteType === 'ETF'; |
|
|
|
|
|
}) |
|
|
|
|
|
.map(({ symbol }) => { |
|
|
|
|
|
return symbol; |
|
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
|
|
const marketData = await this.get(symbols); |
|
|
|
|
|
|
|
|
|
|
|
items = searchResult.quotes |
|
|
|
|
|
.filter((quote) => { |
|
|
|
|
|
return quote.isYahooFinance; |
|
|
|
|
|
}) |
|
|
|
|
|
.filter(({ quoteType }) => { |
|
|
.filter(({ quoteType }) => { |
|
|
return ( |
|
|
return ( |
|
|
quoteType === 'CRYPTOCURRENCY' || |
|
|
quoteType === 'CRYPTOCURRENCY' || |
|
@ -220,40 +205,23 @@ export class YahooFinanceService implements DataProviderInterface { |
|
|
|
|
|
|
|
|
return true; |
|
|
return true; |
|
|
}) |
|
|
}) |
|
|
.map(({ longname, shortname, symbol }) => { |
|
|
.map(({ symbol }) => { |
|
|
return { |
|
|
return symbol; |
|
|
currency: marketData[symbol]?.currency, |
|
|
|
|
|
dataSource: DataSource.YAHOO, |
|
|
|
|
|
name: longname || shortname, |
|
|
|
|
|
symbol: convertFromYahooSymbol(symbol) |
|
|
|
|
|
}; |
|
|
|
|
|
}); |
|
|
}); |
|
|
} catch {} |
|
|
|
|
|
|
|
|
|
|
|
return { items }; |
|
|
const marketData = await this.get(symbols); |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
/** |
|
|
for (const [symbol, value] of Object.entries(marketData)) { |
|
|
* Converts a symbol to a Yahoo symbol |
|
|
items.push({ |
|
|
* |
|
|
symbol, |
|
|
* Currency: USDCHF=X |
|
|
currency: value.currency, |
|
|
* Cryptocurrency: BTC-USD |
|
|
dataSource: DataSource.YAHOO, |
|
|
*/ |
|
|
name: value.name |
|
|
private convertToYahooSymbol(aSymbol: string) { |
|
|
}); |
|
|
if (isCurrency(aSymbol)) { |
|
|
|
|
|
if (isCrypto(aSymbol)) { |
|
|
|
|
|
// Add a dash before the last three characters
|
|
|
|
|
|
// BTCUSD -> BTC-USD
|
|
|
|
|
|
// DOGEUSD -> DOGE-USD
|
|
|
|
|
|
return `${aSymbol.substring(0, aSymbol.length - 3)}-${aSymbol.substring( |
|
|
|
|
|
aSymbol.length - 3 |
|
|
|
|
|
)}`;
|
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
} catch {} |
|
|
|
|
|
|
|
|
return `${aSymbol}=X`; |
|
|
return { items }; |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
return aSymbol; |
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
private parseAssetClass(aPrice: IYahooFinancePrice): { |
|
|
private parseAssetClass(aPrice: IYahooFinancePrice): { |
|
@ -290,7 +258,30 @@ export class YahooFinanceService implements DataProviderInterface { |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
export const convertFromYahooSymbol = (aSymbol: string) => { |
|
|
export const convertFromYahooFinanceSymbol = (aYahooFinanceSymbol: string) => { |
|
|
const symbol = aSymbol.replace('-', ''); |
|
|
const symbol = aYahooFinanceSymbol.replace('-', ''); |
|
|
return symbol.replace('=X', ''); |
|
|
return symbol.replace('=X', ''); |
|
|
}; |
|
|
}; |
|
|
|
|
|
|
|
|
|
|
|
/** |
|
|
|
|
|
* Converts a symbol to a Yahoo Finance symbol |
|
|
|
|
|
* |
|
|
|
|
|
* Currency: USDCHF=X |
|
|
|
|
|
* Cryptocurrency: BTC-USD |
|
|
|
|
|
*/ |
|
|
|
|
|
export const convertToYahooFinanceSymbol = (aSymbol: string) => { |
|
|
|
|
|
if (isCurrency(aSymbol)) { |
|
|
|
|
|
if (isCrypto(aSymbol)) { |
|
|
|
|
|
// Add a dash before the last three characters
|
|
|
|
|
|
// BTCUSD -> BTC-USD
|
|
|
|
|
|
// DOGEUSD -> DOGE-USD
|
|
|
|
|
|
return `${aSymbol.substring(0, aSymbol.length - 3)}-${aSymbol.substring( |
|
|
|
|
|
aSymbol.length - 3 |
|
|
|
|
|
)}`;
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
return `${aSymbol}=X`; |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
return aSymbol; |
|
|
|
|
|
}; |
|
|