setVisibleRange additional argument - right margin % #4512 Indicators are slided when zooming chart #4456 save_load_adapter after removing chart #4454 Overriding timezone definitions #4449 Error while applying study template #4407 Wrong range of overlay after switching timeframe #4372 Intraday Data is Requesting from the server since 1970 #4366 Display price in thousands, millions, billions #4360 setBodyFont of position line tool doesn't work correctly #4353 Realtime updates shifts indicator's data for realtime bars count back #4283 Adaptive indicators dialogs #4247 Click instead of mouseDown for positionLine onClose. #4245 Add applyRightMargin flag to setVisibleRange #4233 Open volume on new pane if volume_force_overlay is enabled #4198 Keep the first visible point when a new bar comes #4188 getVisibleRange should return 00:00 for DWM #4187 21 more indicators #4170 Main series API #4169 Can not set Text property of the Note Shape #4144 Unable to hide legend #4126 setVisibleRange or getVisibleRange adds one more bar to the left #4110 Cannot set tooltip for order line #4079 Dialog is closed immediately if it is invoked from mouse_down event #4077 setVisibleRange doesn't work for multiple charts at once #4068 how to disable mp3 files being loaded? #4052 Increase spacing between bars to display short time periods #4043 Event Marker Placement always uses high #4042 Number cannot be passed as symbol into widget constructor #4039 custom_css_url not working in unstable branch #4017 ConfirmDialog does not close after YES is clicked #3981 Side DOM chart does not render in certain conditions #3975 `getAllShapes` does not return drawings loaded from state #3966 Text inside order/position lines is too small #3962 Removing the last saved chart #3954 get current chart timezone #3943 Text shape throws error setValue #3930 Wrong Symbol and Price values in context menu for secondary instrument. #3926 text override in Horizontal Line does not work #3918 Sticky magnet mode #3902 Add thin bars #3900 Event that study has been removed from the chart #3899 High-Low bars #3898 Multiple Y-axis #3897 Issue with Renko with volume and vwap indicators #3893 Layout rename doesn't send request to server #3878 Cannot create copy of copy of a chart layout #3872 Chart scrolls when tap on a trend line Safari/iphone7 #3871 Chart border overlap scale values #3826 IDatafeedChartApi.subscribeDepth parameters #3821 Error in console when restore defaults #3755 Timeframe is not precise #3722 Wrong year Ticker displayed on X axis if resolution < 2H #3678 New chart layouts #3629 Drawings disappear at certain resolutions #3594 Remove jQuery from loading custom indicator #3563 Align symbol labels #3513 createOrderLine().onMove broken in 1.13 #3480 When changing theme on the fly - chart type changing too #3459 Pivot Points Standard - path to some style params #3441 createMultipointShape overrides with dot don't work #3419 Add inverting price scale #3376 Transparent chart background color #3288 Previous timescale tooltips stay when switching currency #3165 disableSelection still shown the selections on hover #2864 Override symbol from saved_data #2493 Forecast balloon too short #2289 Context menu submenus cover up other options #2007 Add setVisiblePriceRange method #1408 Modify panes height/order #1232 Add custom Interval #1191
275 lines
9.5 KiB
TypeScript
275 lines
9.5 KiB
TypeScript
import {
|
|
LibrarySymbolInfo,
|
|
SearchSymbolResultItem,
|
|
} from '../../../charting_library/datafeed-api';
|
|
|
|
import {
|
|
getErrorMessage,
|
|
logMessage,
|
|
} from './helpers';
|
|
|
|
import { Requester } from './requester';
|
|
|
|
interface SymbolInfoMap {
|
|
[symbol: string]: LibrarySymbolInfo | undefined;
|
|
}
|
|
|
|
interface ExchangeDataResponseSymbolData {
|
|
'type': string;
|
|
'timezone': LibrarySymbolInfo['timezone'];
|
|
'description': string;
|
|
|
|
'exchange-listed': string;
|
|
'exchange-traded': string;
|
|
|
|
'session-regular': string;
|
|
|
|
'fractional': boolean;
|
|
|
|
'pricescale': number;
|
|
|
|
'ticker'?: string;
|
|
|
|
'minmov2'?: number;
|
|
'minmove2'?: number;
|
|
|
|
'minmov'?: number;
|
|
'minmovement'?: number;
|
|
|
|
'force-session-rebuild'?: boolean;
|
|
|
|
'supported-resolutions'?: string[];
|
|
'intraday-multipliers'?: string[];
|
|
|
|
'has-intraday'?: boolean;
|
|
'has-daily'?: boolean;
|
|
'has-weekly-and-monthly'?: boolean;
|
|
'has-empty-bars'?: boolean;
|
|
'has-no-volume'?: boolean;
|
|
|
|
'volume-precision'?: number;
|
|
}
|
|
|
|
// Here is some black magic with types to get compile-time checks of names and types
|
|
type PickArrayedObjectFields<T> = Pick<T, {
|
|
// tslint:disable-next-line:no-any
|
|
[K in keyof T]-?: NonNullable<T[K]> extends any[] ? K : never;
|
|
}[keyof T]>;
|
|
|
|
type ExchangeDataResponseArrayedSymbolData = PickArrayedObjectFields<ExchangeDataResponseSymbolData>;
|
|
type ExchangeDataResponseNonArrayedSymbolData = Pick<ExchangeDataResponseSymbolData, Exclude<keyof ExchangeDataResponseSymbolData, keyof ExchangeDataResponseArrayedSymbolData>>;
|
|
|
|
type ExchangeDataResponse =
|
|
{
|
|
symbol: string[];
|
|
} &
|
|
{
|
|
[K in keyof ExchangeDataResponseSymbolData]: ExchangeDataResponseSymbolData[K] | NonNullable<ExchangeDataResponseSymbolData[K]>[];
|
|
};
|
|
|
|
function extractField<Field extends keyof ExchangeDataResponseNonArrayedSymbolData>(data: ExchangeDataResponse, field: Field, arrayIndex: number): ExchangeDataResponseNonArrayedSymbolData[Field];
|
|
function extractField<Field extends keyof ExchangeDataResponseArrayedSymbolData>(data: ExchangeDataResponse, field: Field, arrayIndex: number, valueIsArray: true): ExchangeDataResponseArrayedSymbolData[Field];
|
|
function extractField<Field extends keyof ExchangeDataResponseSymbolData>(data: ExchangeDataResponse, field: Field, arrayIndex: number, valueIsArray?: boolean): ExchangeDataResponseSymbolData[Field] {
|
|
const value: ExchangeDataResponse[keyof ExchangeDataResponseSymbolData] = data[field];
|
|
|
|
if (Array.isArray(value) && (!valueIsArray || Array.isArray(value[0]))) {
|
|
return value[arrayIndex];
|
|
}
|
|
|
|
return value as ExchangeDataResponseSymbolData[Field];
|
|
}
|
|
|
|
export class SymbolsStorage {
|
|
private readonly _exchangesList: string[] = ['NYSE', 'FOREX', 'AMEX'];
|
|
private readonly _symbolsInfo: SymbolInfoMap = {};
|
|
private readonly _symbolsList: string[] = [];
|
|
private readonly _datafeedUrl: string;
|
|
private readonly _readyPromise: Promise<void>;
|
|
private readonly _datafeedSupportedResolutions: string[];
|
|
private readonly _requester: Requester;
|
|
|
|
public constructor(datafeedUrl: string, datafeedSupportedResolutions: string[], requester: Requester) {
|
|
this._datafeedUrl = datafeedUrl;
|
|
this._datafeedSupportedResolutions = datafeedSupportedResolutions;
|
|
this._requester = requester;
|
|
this._readyPromise = this._init();
|
|
this._readyPromise.catch((error: Error) => {
|
|
// seems it is impossible
|
|
// tslint:disable-next-line:no-console
|
|
console.error(`SymbolsStorage: Cannot init, error=${error.toString()}`);
|
|
});
|
|
}
|
|
|
|
// BEWARE: this function does not consider symbol's exchange
|
|
public resolveSymbol(symbolName: string): Promise<LibrarySymbolInfo> {
|
|
return this._readyPromise.then(() => {
|
|
const symbolInfo = this._symbolsInfo[symbolName];
|
|
if (symbolInfo === undefined) {
|
|
return Promise.reject('invalid symbol');
|
|
}
|
|
|
|
return Promise.resolve(symbolInfo);
|
|
});
|
|
}
|
|
|
|
public searchSymbols(searchString: string, exchange: string, symbolType: string, maxSearchResults: number): Promise<SearchSymbolResultItem[]> {
|
|
interface WeightedItem {
|
|
symbolInfo: LibrarySymbolInfo;
|
|
weight: number;
|
|
}
|
|
|
|
return this._readyPromise.then(() => {
|
|
const weightedResult: WeightedItem[] = [];
|
|
const queryIsEmpty = searchString.length === 0;
|
|
|
|
searchString = searchString.toUpperCase();
|
|
|
|
for (const symbolName of this._symbolsList) {
|
|
const symbolInfo = this._symbolsInfo[symbolName];
|
|
|
|
if (symbolInfo === undefined) {
|
|
continue;
|
|
}
|
|
|
|
if (symbolType.length > 0 && symbolInfo.type !== symbolType) {
|
|
continue;
|
|
}
|
|
|
|
if (exchange && exchange.length > 0 && symbolInfo.exchange !== exchange) {
|
|
continue;
|
|
}
|
|
|
|
const positionInName = symbolInfo.name.toUpperCase().indexOf(searchString);
|
|
const positionInDescription = symbolInfo.description.toUpperCase().indexOf(searchString);
|
|
|
|
if (queryIsEmpty || positionInName >= 0 || positionInDescription >= 0) {
|
|
const alreadyExists = weightedResult.some((item: WeightedItem) => item.symbolInfo === symbolInfo);
|
|
if (!alreadyExists) {
|
|
const weight = positionInName >= 0 ? positionInName : 8000 + positionInDescription;
|
|
weightedResult.push({ symbolInfo: symbolInfo, weight: weight });
|
|
}
|
|
}
|
|
}
|
|
|
|
const result = weightedResult
|
|
.sort((item1: WeightedItem, item2: WeightedItem) => item1.weight - item2.weight)
|
|
.slice(0, maxSearchResults)
|
|
.map((item: WeightedItem) => {
|
|
const symbolInfo = item.symbolInfo;
|
|
return {
|
|
symbol: symbolInfo.name,
|
|
full_name: symbolInfo.full_name,
|
|
description: symbolInfo.description,
|
|
exchange: symbolInfo.exchange,
|
|
params: [],
|
|
type: symbolInfo.type,
|
|
ticker: symbolInfo.name,
|
|
};
|
|
});
|
|
|
|
return Promise.resolve(result);
|
|
});
|
|
}
|
|
|
|
private _init(): Promise<void> {
|
|
interface BooleanMap {
|
|
[key: string]: boolean | undefined;
|
|
}
|
|
|
|
const promises: Promise<void>[] = [];
|
|
const alreadyRequestedExchanges: BooleanMap = {};
|
|
|
|
for (const exchange of this._exchangesList) {
|
|
if (alreadyRequestedExchanges[exchange]) {
|
|
continue;
|
|
}
|
|
|
|
alreadyRequestedExchanges[exchange] = true;
|
|
promises.push(this._requestExchangeData(exchange));
|
|
}
|
|
|
|
return Promise.all(promises)
|
|
.then(() => {
|
|
this._symbolsList.sort();
|
|
logMessage('SymbolsStorage: All exchanges data loaded');
|
|
});
|
|
}
|
|
|
|
private _requestExchangeData(exchange: string): Promise<void> {
|
|
return new Promise((resolve: () => void, reject: (error: Error) => void) => {
|
|
this._requester.sendRequest<ExchangeDataResponse>(this._datafeedUrl, 'symbol_info', { group: exchange })
|
|
.then((response: ExchangeDataResponse) => {
|
|
try {
|
|
this._onExchangeDataReceived(exchange, response);
|
|
} catch (error) {
|
|
reject(error);
|
|
return;
|
|
}
|
|
|
|
resolve();
|
|
})
|
|
.catch((reason?: string | Error) => {
|
|
logMessage(`SymbolsStorage: Request data for exchange '${exchange}' failed, reason=${getErrorMessage(reason)}`);
|
|
resolve();
|
|
});
|
|
});
|
|
}
|
|
|
|
private _onExchangeDataReceived(exchange: string, data: ExchangeDataResponse): void {
|
|
let symbolIndex = 0;
|
|
|
|
try {
|
|
const symbolsCount = data.symbol.length;
|
|
const tickerPresent = data.ticker !== undefined;
|
|
|
|
for (; symbolIndex < symbolsCount; ++symbolIndex) {
|
|
const symbolName = data.symbol[symbolIndex];
|
|
const listedExchange = extractField(data, 'exchange-listed', symbolIndex);
|
|
const tradedExchange = extractField(data, 'exchange-traded', symbolIndex);
|
|
const fullName = tradedExchange + ':' + symbolName;
|
|
|
|
const ticker = tickerPresent ? (extractField(data, 'ticker', symbolIndex) as string) : symbolName;
|
|
|
|
const symbolInfo: LibrarySymbolInfo = {
|
|
ticker: ticker,
|
|
name: symbolName,
|
|
base_name: [listedExchange + ':' + symbolName],
|
|
full_name: fullName,
|
|
listed_exchange: listedExchange,
|
|
exchange: tradedExchange,
|
|
description: extractField(data, 'description', symbolIndex),
|
|
has_intraday: definedValueOrDefault(extractField(data, 'has-intraday', symbolIndex), false),
|
|
has_no_volume: definedValueOrDefault(extractField(data, 'has-no-volume', symbolIndex), false),
|
|
minmov: extractField(data, 'minmovement', symbolIndex) || extractField(data, 'minmov', symbolIndex) || 0,
|
|
minmove2: extractField(data, 'minmove2', symbolIndex) || extractField(data, 'minmov2', symbolIndex),
|
|
fractional: extractField(data, 'fractional', symbolIndex),
|
|
pricescale: extractField(data, 'pricescale', symbolIndex),
|
|
type: extractField(data, 'type', symbolIndex),
|
|
session: extractField(data, 'session-regular', symbolIndex),
|
|
timezone: extractField(data, 'timezone', symbolIndex),
|
|
supported_resolutions: definedValueOrDefault(extractField(data, 'supported-resolutions', symbolIndex, true), this._datafeedSupportedResolutions),
|
|
force_session_rebuild: extractField(data, 'force-session-rebuild', symbolIndex),
|
|
has_daily: definedValueOrDefault(extractField(data, 'has-daily', symbolIndex), true),
|
|
intraday_multipliers: definedValueOrDefault(extractField(data, 'intraday-multipliers', symbolIndex, true), ['1', '5', '15', '30', '60']),
|
|
has_weekly_and_monthly: extractField(data, 'has-weekly-and-monthly', symbolIndex),
|
|
has_empty_bars: extractField(data, 'has-empty-bars', symbolIndex),
|
|
volume_precision: definedValueOrDefault(extractField(data, 'volume-precision', symbolIndex), 0),
|
|
format: 'price',
|
|
};
|
|
|
|
this._symbolsInfo[ticker] = symbolInfo;
|
|
this._symbolsInfo[symbolName] = symbolInfo;
|
|
this._symbolsInfo[fullName] = symbolInfo;
|
|
|
|
this._symbolsList.push(symbolName);
|
|
}
|
|
} catch (error) {
|
|
throw new Error(`SymbolsStorage: API error when processing exchange ${exchange} symbol #${symbolIndex} (${data.symbol[symbolIndex]}): ${error.message}`);
|
|
}
|
|
}
|
|
}
|
|
|
|
function definedValueOrDefault<T>(value: T | undefined, defaultValue: T): T {
|
|
return value !== undefined ? value : defaultValue;
|
|
}
|