**Knowledge Prerequisite* – Must have an understanding of a Refinitiv streaming API such as EMA and be familiar with consuming OMM Market Price data (see end of article for links to relevant tutorials)*
Developers often ask 'how can I get a list of all the RICs available?' - without realising that our Refinitiv Real-time feed carries millions of instruments (over 80 million at the time of writing). Instruments expire and new ones are added all the time. So, if they were to get a list of all the RICs available, the volume would no doubt be overwhelming.
However, the Refinitiv Real-Time Edge Device (see the API_Concepts Guide if you are not familiar with the Edge device) does have a facility that allows the developer to make a Criteria Based Request (CBR) for a list of RICs matching a set of specified criteria - also known as a BDS (Broadcast Data Stream).
For example, you might want to discover:
- a list of all Treasury Debt Futures Instruments for the Eurex Exchange
- or you may want all the RICs for Cash Options and Futures Options for the Hong Kong Exchange
- or even a list of all the RICs from all the APAC exchanges
From a developer's perspective, your interface for utilising CBR / BDS functionality is the OMM SymbolList domain. You can read in more detail about OMM SymbolLists in the Developer Guides or API Concept Guides that come with our Refinitiv Real-time and RFA API packages. However, the following should provide a quick overview.
SymbolList domain
The SymbolList domain is the OMM replacement for the legacy Chains mechanism (see AboutChains for more details). This is where the application makes a SymbolList Domain request using a SymbolList name (in place of a RIC Code) and the server responds with a Map structure containing the corresponding RICs.
So, for example if I were to make a SymbolList request for '.AV.O' (Top 25 most active RICs on Nasdaq) from Refinitiv Real-timeI would receive back something like this:
As you can see a list of RICs is returned as the Key value for each Map entry (I am using the RFA Java OMM Viewer example for the purpose of this tutorial as it provides a convenient GUI interface). You will note that along with the RIC as the Map Key, I get back a RANK_POS field - as it makes sense in the context of this particular SymbolList (i.e. top 25). For an actual CBR / BDS response you will not get this field (see later).
Just to be clear '.AV.O' is not a CBR / BDS - it just happens to be a SymbolList name that is supported by the Refinitiv Real-time data service.
Now that I have summarised SymbolList, I can move on to explaining the CBR / BDS mechanism.
Defining your RIC selection criteria
I mentioned earlier that you can define a Criteria Based Request by specifying a set of values to match against. How do you go about doing this and what types of Criteria can you specify?
Technical Prerequisites
To define the match criteria you will need access to the Refinitiv Real-time Edge console application. So, the first thing you need to do is speak to your Market Data team and confirm that they will be able to create what is known as a BDS Term on your organisation's Edge devices.
Creating a BDS Term
A BDS Term is made up of one or more SQL Terms, other BDS Terms and also File Terms.
For the subject of this article i.e. discovering a list of RICs matching some selection Criteria, the SQL Term is the most appropriate. Using a File Term (a file populated with a list of RICs) would require prior knowledge - so does not make sense in terms of discovering RICs.
Creating a SQL Term
A SQL Term is a SQL statement that is used to query the Refinitiv Real-time Edge database. The resulting RIC list obtained from the query is used as the response for the criteria based request.
There a limited set of fields that can be used to form the SQL query for example:
SQL Field |
User Input Type |
Notes |
---|---|---|
Exchange ID |
Numeric value |
Values can be found in the enumtype.def file that comes with each TR API - in the Enum definitions for RDN_EXCHD2 and RDN_EXCHID |
Exchange ID (Name) |
Text based equivalent |
Can be selected using the SQL query editor (see below) |
Record Type |
Numeric value |
Type of record e.g. Debt, Futures, Options etc (see link below) |
Record Type (Name) |
Text based equivalent |
Type of record e.g. Debt, Futures, Options etc (see link below) |
RIC Name |
Text |
Wildcard characters as detailed in table below |
You can find an explanation of Record Types in the Article about Using FID259 (RECORDTYPE) on this website
RIC Name Wildcards
The RIC Name matching criteria can use the following Wildcarding symbols:
Description |
Character |
Example |
Meaning |
Single Character |
'_' |
IBM._ |
All RICs with “IBM.” as prefix and followed by any single characters. e.g. IBM.O, IBM.N, IBM.t |
Multiple characters |
'%' |
%.HK |
All RICs from HK exchange with “.HK” as suffix |
Range |
[] |
800[2-6].HK |
Represents RICs 8002.HK, 8003.HK, 8004.HK, 8005.HK, and 8006.HK |
Negate |
'^' |
_[^#]% |
All RICs that are not chain records (i.e. not 0#GC:, 0#.FTSE, 1#.HSI, etc) |
Using the Edge console
Access to the Edge console is usually limited to your site's Market Data admin team. However, if they are not familiar with its use, the following should help.
BDS terms and SQL terms can be created via the Task menu of the Edge console application.
Since your BDS Term is going to be made up of one or more SQL Terms, it makes sense to Create your SQL Terms first.
The above shows the fuller set of Criteria / fields you can match on when constructing your SQL term.
So, let us go ahead and create a SQL Term:
In the above example I have a SQL query which will select all RICs from ICE(758), Amsterdam Exchanges(77) and the Stockholm Exchange (70)
Next, I am going to save the SQL Term as 'SQL_EUROPE_V1' :
Now that I have a SQL Term created I can go ahead and Create the BDS Term that will use the SQL Term:
Here I can select the terms that will make up my new BDS Term. I just going to select the SQL_EUROPE_V1 Term that I created above. However, as you can see it is possible to select multiple SQL Terms as well as other existing BDS Terms - allowing you to make your selection criteria as simple or as complex as required.
Finally, I am going to save this BDS Term as BDS_EUROPE_V1 :
So, now I have my SQL Term and a BDS Term that references the SQL Term.
I have kept things simple, however, as explained above you can use multiple SQL Terms within a single BDS Term e.g. the following BDS Term is made of 3 SQL Terms:
Testing the BDS Term
To test the new BDS Term we need an application which can request and process a SymbolList Domain. All of our OMM capable API packages include a simple SymbolList example including the Enterprise Message API - which should be the shortest in terms of lines of code. So, for brevity I will use the EMA Java example ex270_SymbolList.
Before, I proceed I should point out that if I made a request for the above BDS_EUROPE_V1 Term, it would return somewhere in the region 740,000 instruments - not ideal for testing with. In view of this I created another SQL term with a much smaller range of matches as follows:
Requesting a BDS
As mentioned above I am going to use ex270_SymbolList which can be found in the Java\Ema\Examples\src\main\java\com\refinitiv\ema\examples\training\consumer\series200\ folder of the Java RT-SDK package (a similar 270_SymbolList example is also available in the C++ version of the RT-SDK).
When requesting the more commonly used MarketPrice data you specify a Domain of MARKET_PRICE in the Request Messsage e.g.
RFA : ommmsg.setMsgModelType(RDMMsgTypes.MARKET_PRICE)
EMA : reqMsg.domainType(EmaRdm.MMT_MARKET_PRICE) // default domain in EMA - so not explicitly required
However, if you look at the main() method of example270 you will note the domain for the request message is specified as MMT_SYMBOL_LIST when creating the ReqMsg :
consumer.registerClient(EmaFactory.createReqMsg().domainType(EmaRdm.MMT_SYMBOL_LIST)
.serviceName("ELEKTRON_DD").name("BDS_VOD"), appClient, 0);
You will also note that I have changed the name of the of the SymbolList that I am requesting to BDS_VOD.
Another key difference between MarketPrice data and a SymbolList is that the SymbolList response is in the form of a Map - rather than the simpler FieldList of the MarketPrice response.
If you are familiar with OMM Map you can skip the following section and move onto the Results.
You can read in more detail about OMM Map in the Developer Guides or API Concept Guides that come with our RT-SDK and RFA API packages. However, the following should provide a quick overview:
An example of a SymbolList Refresh response trace below (truncated) should help understand the Map's structure better:
<REFRESH domainType="SYMBOL_LIST" streamId="5" containerType="MAP" flags="0x1F8
(HAS_MSG_KEY|SOLICITED|REFRESH_COMPLETE)" State: Open/Ok/None - text: "All is well">
<key serviceId="257" name="BDS_VOD"/>
<dataBody>
<map countHint="30" containerType="FIELD_LIST" >
<summaryData>
<fieldList>
<fieldEntry fieldId="6456" data="06"/>
</fieldList>
</summaryData>
<mapEntry flags="0x00" action="ADD" key="VOD.BN" >
<fieldList>
<fieldEntry fieldId="1" data="0D42"/>
</fieldList>
</mapEntry>
<mapEntry flags="0x00" action="ADD" key="VOD.MW" >
<fieldList>
<fieldEntry fieldId="1" data="19A1"/>
</fieldList>
</mapEntry>
<mapEntry flags="0x00" action="ADD" key="VOD.SId" >
<fieldList>
<fieldEntry fieldId="1" data="1AD0"/>
</fieldList>
</mapEntry>
....
....
<mapEntry flags="0x00" action="ADD" key="VOD.SI" >
<fieldList>
<fieldEntry fieldId="1" data="0BEB"/>
</fieldList>
</mapEntry>
</map>
</dataBody>
</REFRESH>
A breakdown of the above Refresh Message snippet:
- First we have the 'header' section which confirms various attributes of the Refresh including domain, unique stream ID, Status information, name and the container type
- Next we have have the Map payload which consists of a Summary section - not particularly useful in the case of a BDS response - just the Domain type repeated again. However, for something like a Level 2 OrderBook (which also uses a Map), the Summary would contain details like Currency, Trade Units, Exchange ID - i.e. values that are common to all the orders in the book
- Then we have the individual Map entries
- The action of Add indicates this is a new entry that needs to be added to your local cache - i.e. your in-memory local representation of the Map
- If this was an Update Message, it may contain entries with an action of Delete - indicating the item has expired and should be deleted from local cache (for an OrderBook you can also receive Update actions e.g. if the price or size of an order book entry changes)
- The Map Key for a BDS SymbolList entry is the actual RIC itself (for an OrderBook the Key could be the Order Price + Order Side)
- The Field List contains one ore more fields relevant to the Map Entry. Here we just have a permissioning code for the RIC (for an Order Book entry it would the Price, side, size, time of order etc).
Result of BDS SymbolList Request
Once I had defined the SQL_VOD Term and linked it to the BDS_VOD term I then requested it using sample Consumer application. I made a SymbolList Request for BDS_VOD from the ELEKTRON_DD service and the result was as shown below:
The service name may vary for your site - it needs to be the service which delivers the Refinitiv Real-Time data feed - often named IDN_RDF, ELEKTRON_DD or something similar.
As the number of RICs returned above was quite small, the complete Map arrived in a single Refresh Message. A more realistic BDS query will most likely contain a much high number of RICs and therefore these will be delivered as multi-part RefreshMsgs, where each RefreshMsg contains a subset of the RICs. The final RefreshMsg will have the Complete flag set to true, indicating that all parts have been sent and the Order Book delivery is now complete.
You will note the REFRESH_COMPLETE flag in the REFRESH trace above - as this was single self contained RefreshMsg. You can test for the final RefreshMsg in code by testing for the complete flag e.g.
public void onRefreshMsg(RefreshMsg refreshMsg, OmmConsumerEvent event)
{
...
if (refreshMsg.complete())
System.out.println("Final RefreshMsg received");
...
}
Note: You can make a Snapshot request for a BDS and you will receive the current list of matching RICs and the data stream will be closed off. If, however, you make a Streaming request, the stream will remain open (until you close it). You will receive Update Messages containing Map Entries with:
- Add action when any new RICs matching your Criteria become available
- Delete action when any of the previously matched RICs expire
- Update action when the permission code for existing RICs change (a rare occurence)
Often the Update message will contain a Map with a mixture of Add & Delete action entries - as RICs expire and new RICs are created at certain points during the day.
Use cases
I have come across two real life use cases for the BDS / CBR functionality.
- Maintaining a golden source - a SymbolList consumer is run in snapshot mode several times. The runs take place shortly after midnight - invoking a different BDS each time covering various exchanges. The application then performs a delta with an existing repository to arrive at a list of new and expired RICs. The new and expired lists are then used to update the golden source. Market Data for the RICs themselves is not requested at this time - the golden source is used later during market open times to request Market data for required items.
- Obtaining snapshots during the day - a SymbolList consumer is run in snapshot mode at two points during the day. Once the complete list of RICs has been obtained, the application then makes MarketPrice snapshot requests for each RIC and the Market Data is captured.
The 1st use case makes the BDS request during non-trading hours mainly because the total number of RICs received runs into a few million and processing them takes time. Also, the daytime requirement is to request Market Data for only a fraction of the RICs in the golden source.
In comparison, the 2nd use case involves only a few thousand RICs and the requirement is to capture Market Data for all the RICs returned in the SymbolList.
Requesting Market Price Data for the RICs
If your use case involves requesting Market Price domain Data for all the RICs returned in the SymboList, there is a feature of the SymbolList domain which you could utilise to automate the requesting of all the RICs.
The SymbolList section of the RDM Usage Guide (that comes with each API) describes Symbol List Behaviours in more detail. Below is a snippet from the document:
Element Name |
Type |
Default |
Description |
SymbolListBehaviors |
ElementList |
ElementList containing ElementEntry of :DataStreams set to 0. |
Indicates any expected data behavior of individual items that will be opened from the symbol list. If this element is absent, individual streams will not be opened. |
The :DataStreams element entry can be one of the following values
Value |
Result |
0x0 |
The consumer is interested only in getting the names and no data on the individual items of the symbol list. This is the default behavior. |
0x1 |
The consumer is interested in getting the individual items of the symbol list opened as streaming. |
0x2 |
The consumer is interested in getting the individual items of the symbol list opened as snapshots. |
So, for example if I modify the main() method of the above example to include the following, the example will request SymbolList and also receive MarketPrice snapshot data for each of the RICs in the list.
public static void main(String[] args)
{
....
ElementList symbolListBehaviors= EmaFactory.createElementList();
ElementList dataStream= EmaFactory.createElementList();
dataStream.add(EmaFactory.createElementEntry().uintValue(EmaRdm.ENAME_DATA_STREAMS,
SymbolListDataStreamRequestFlags.SYMBOL_LIST_DATA_SNAPSHOTS));
symbolListBehaviors.add(EmaFactory.createElementEntry()
.elementList(EmaRdm.ENAME_SYMBOL_LIST_BEHAVIORS, dataStream));
...
...
consumer.registerClient(EmaFactory.createReqMsg().domainType(EmaRdm.MMT_SYMBOL_LIST)
.serviceName("ELEKTRON_DD").name("BDS_VOD").payload(symbolListBehaviors),
appClient, 0);
...
}
As you can see I am creating an ElementList with entry of DataStream, specifying Data Snapshot as the value, and then I add this to the SymbolListBehaviours ElementList. I then attach this as the Payload of the BDS SymbolList Request Message.
I also modified the onRefreshMsg() handler to process FIELD_LIST rather than just MAP payloads, so that the Market Price data is also output to the console when received.
public void onRefreshMsg(RefreshMsg refreshMsg, OmmConsumerEvent event)
{
...
if (DataType.DataTypes.MAP == refreshMsg.payload().dataType())
decode(refreshMsg.payload().map());
if (DataType.DataTypes.FIELD_LIST == refreshMsg.payload().dataType())
decode(refreshMsg.payload().fieldList(), false);
...
}
When I re-ran the modified example, I received back the BDS SymbolList response and a snapshot for each of the items in the SymbolList.
This data auto-request facility is not BDS specific and so can be used generally with SymbolList requests.
Closing summary
To summarise the information above:
- The Refinitiv Real-Time EDGE device offers a realtime RIC discovery facility
- To use this facility you need access to the Edge Console application
- The facility is referred to as Broadcast Data Stream - BDS (also known as Criteria Based Request - CBR)
- BDS Terms can be made up of SQL Terms, File Terms and include other BDS terms
- An OMM Consumer application needs to make a SymbolList domain request using the defined BDS term name as the RIC name
- The response is sent as single or multi-part Refresh Messages (final Refresh Message has the Complete flag set to true)
- The payload consists of a Map, where each Map entry's Key is the actual RIC itself and each entry also has an action; Add, Delete or Update
- A Snapshot request will deliver a list of RICs that match the SQL Terms criteria at that point in time
- A Streaming request will continue to receive Update Messages as and when RICs expire or new RICs are created
- It also possible to auto-request Snapshot or Streaming data for each RIC in the SymbolList by utilising the Symbol List Behaviours
Additional Resources
You will find links to the APIs, Tutorials, Helpdesk and Q&A Forum in the Links Panel