API Documentation

NxMSG_CATEGORY message

The second parameter to the NxCoreCallback function, NxCoreMessage, holds the symbol and session information in pNxCoreMessage->coreHeader, and data information in pNxCoreMessage->coreData.Category, which is a NxCoreCategory data structure.

Sent for all other data types, such as fundamental information, 52 week/contract/YTD high/low data, additional trade correction information, exchange code translations, trade/quote condition code translation, trading halts, trading imbalances, open trade indications, etc. A typical trading day will have 1-2 million or so of these message types, with the bulk of them occurring at the start of each tape summarizing the previous trading session.

Importance of SessionDate

It is critical that you look at the pNxCoreMessage->coreHeader.nxSessionDate for every Category, and ensure it's the date you need. Some category messages come in multiple times per day, for the same symbol -- but the data in them is for different dates. Good example of that is the OpenInterest category.

A simple (but incomplete) way of checking if something is "too old" is something like this:

    // if this message's is for a session that's over a week old, skip it
    if (pNxCoreMessage->coreHeader.nxSessionDate.NDays + 7 < pNxCoreSys->nxDate.NDays)
    {
        // skip
    }

Importance of NxCategoryField.Set field

It is critical that you look at the NxCategoryField.Set value for every Category field, and ensure it's set to 1 to make sure the data for that field is present in the current Category message. Some category messages come in multiple times per day, for the same symbol -- but the data in them contains only the fields that changed and not every field in the entire category.

See the code example at the bottom of this page that shows a good technique for accomplishing this.

Simplest example

Here's the simplest example.

#include <NxCoreAPI.h>
int processNxCoreCategory(const NxCoreSystem* pNxCoreSys, const NxCoreMessage* pNxCoreMsg)
{
    return NxCALLBACKRETURN_CONTINUE;
}
int __stdcall OnNxCoreCallback(const NxCoreSystem* pNxCoreSys, const NxCoreMessage* pNxCoreMsg)
{
    switch( pNxCoreMsg->MessageType ) {
        case NxMSG_STATUS:       break;
        case NxMSG_EXGQUOTE:     break;
        case NxMSG_MMQUOTE:      break;
        case NxMSG_TRADE:        break;
        case NxMSG_CATEGORY:     return processNxCoreCategory(pNxCoreSys, pNxCoreMsg);
        case NxMSG_SYMBOLCHANGE: break;
        case NxMSG_SYMBOLSPIN:   break;
    }
    return NxCALLBACKRETURN_CONTINUE;
}

Example

Here's an example category dumper that reflectively pretty-prints out every category

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include "NxCoreAPI.h"
#include "NxCoreAPI_Wrapper_C++.h"

NxCoreClass NxCore;

void categoryDump(const NxCoreSystem* pNxCoreSys, const NxCoreMessage* pNxCoreMessage)
{
    const NxCoreHeader   &ch  = pNxCoreMessage->coreHeader;
    const NxCoreCategory &cat = pNxCoreMessage->coreData.Category;

    char symbol[128] = {0};

    if (ch.pnxOptionHdr)
    {
        if (ch.pnxOptionHdr->pnxsDateAndStrike)
        {
            // If pnxsDateAndStrike->String[0] == ' ', then this symbol is in new OSI format. 
            if (ch.pnxOptionHdr->pnxsDateAndStrike->String[0]==' ')
            {
                sprintf(symbol,"%s%02d%02d%02d%c%08d",
                    ch.pnxStringSymbol->String,
                    ch.pnxOptionHdr->nxExpirationDate.Year-2000,
                    ch.pnxOptionHdr->nxExpirationDate.Month,
                    pNxCoreMessage->coreHeader.pnxOptionHdr->nxExpirationDate.Day,
                    (ch.pnxOptionHdr->PutCall == 0) ? 'C' : 'P',
                    ch.pnxOptionHdr->strikePrice);
            }
            // Otherwise the symbol is in old OPRA format.
            else
            {
                sprintf(symbol,"%s%c%c",
                    ch.pnxStringSymbol->String,
                    ch.pnxOptionHdr->pnxsDateAndStrike->String[0],
                    ch.pnxOptionHdr->pnxsDateAndStrike->String[1]);
            }
        }
        else
            sprintf(symbol,"%s", ch.pnxStringSymbol->String);
    }

    printf("Category         : %s(%ld)\n",          cat.pnxStringCategory->String, cat.pnxStringCategory->Atom);
    printf("  nxTime         : %02d:%02d:%02d\n",   (int) pNxCoreSys->nxTime.Hour, (int) pNxCoreSys->nxTime.Minute, (int) pNxCoreSys->nxTime.Second);
    printf("  symbol         : %s\n",               symbol);
    printf("  listedExch     : %ld\n",              ch.ListedExg);
    printf("  sessionDate    : %d%02d%02d\n",       (int) ch.nxSessionDate.Year, (int) ch.nxSessionDate.Month, (int) ch.nxSessionDate.Day);
    printf("  time           : %02d:%02d:%02d\n",   (int) ch.nxExgTimestamp.Hour, (int) ch.nxExgTimestamp.Minute, (int) ch.nxExgTimestamp.Second);
    printf("  reportExch     : %ld\n",              ch.ReportingExg);
    printf("  sessionId      : %ld\n",              (int) ch.SessionID);
    printf("  permissionId   : %ld [%s]\n",         ch.PermissionID, NxCore.GetDefinedString(NxST_PERMID, ch.PermissionID));
    printf("  NFields        : %ld\n",              cat.NFields);

    for (int i = 0; i < cat.NFields; i++)
    {
        const NxCategoryField& cf = cat.pnxFields[i];

        printf("     %ld|%s|[%s] {%ld} :", i, cf.FieldName, cf.FieldInfo, (int) cf.Set);

        if (!cf.Set)
        {
            printf("  -- !Set --\n");
            continue;
        }

        switch (cf.FieldType)
        {
            case NxCFT_UNKNOWN:
            {
                printf("  -- unknown --");
                break;
            }
            case NxCFT_64BIT:
            {
                printf("  %I64d", cf.data.i64Bit);
                break;
            }
            case NxCFT_32BIT:
            {
                printf("  %ld", cf.data.i32Bit);
                break;
            }
            case NxCFT_STRINGZ:
            {
                printf("  %s", cf.data.StringZ);
                break;
            }
            case NxCFT_DOUBLE:
            {
                printf("  %lf", cf.data.Double);
                break;
            }
            case NxCFT_PRICE:
            {
                printf("  %.2lf", NxCore.PriceToDouble(cf.data.nxPrice.Price, cf.data.nxPrice.PriceType));
                break;
            }
            case NxCFT_DATE:
            {
                printf("  %d%02d%02d",  (int) cf.data.nxDate.Year, (int) cf.data.nxDate.Month, (int) cf.data.nxDate.Day);
                break;
            }
            case NxCFT_TIME:
            {
                printf("  %02d:%02d:%02d.%03d", (int) cf.data.nxTime.Hour, (int) cf.data.nxTime.Minute, (int) cf.data.nxTime.Second, (int) cf.data.nxTime.Millisecond);
                break;
            }
            case NxCFT_NxSTRING:
            {
                if (cf.data.pnxString)
                {
                    const NxString& s = (const NxString&) *cf.data.pnxString;
                    printf("  '%s'", s.String);
                }
                break;
            }
            case NxCFT_STRING_IDX:
            {
                const StringTableItem& stItem = (const StringTableItem&) cf.data.stringTableItem;
                const char* ss = NxCore.GetDefinedString(stItem.ixTable, stItem.idString);
                if (!ss)
                {
                    ss = "***NULL***";
                }
                printf("  [%ld:%ld] '%s'", stItem.ixTable, stItem.idString, ss);
                break;
            }
            case NxCFT_STRING_MAP:
            {
                const StringTableItem& stItem = (const StringTableItem&) cf.data.stringTableItem;
                bool printed = false;
                unsigned int id = stItem.idString;
                for (int iMap = 0; id; ++iMap, id >>= 1)
                {
                    if (id & 1)
                    {
                        const char* ss = NxCore.GetDefinedString(stItem.ixTable, iMap);
                        if (!ss)
                        {
                            ss = "***NULL***";
                        }
                        if (!printed)
                        {
                            printf("  [%ld:%ld] '%s'", stItem.ixTable, iMap, ss);
                            printed = true;
                        }
                        else
                        {
                            printf(",  [%ld:%ld] '%s'", stItem.ixTable, iMap, ss);
                        }
                    }
                }
                break;
            }
        }

        printf("\n");
    }
}

int __stdcall nxCoreCallback(const NxCoreSystem* pNxCoreSys, const NxCoreMessage* pNxCoreMessage)
{
    switch (pNxCoreMessage->MessageType)
    {
        case NxMSG_CATEGORY:
        {
            categoryDump(pNxCoreSys, pNxCoreMessage);
            break;
        }
    }

    return NxCALLBACKRETURN_CONTINUE;
}

int main(int argc, char** argv)
{
    if (!NxCore.LoadNxCore("NxCoreAPI64.dll") &&
        !NxCore.LoadNxCore("NxCoreAPI.dll"))
    {
        fprintf(stderr, "loading library failed\n");
        return -1;
    }
    NxCore.ProcessTape(argv[1], 0, NxCF_EXCLUDE_CRC_CHECK | NxCF_EXCLUDE_QUOTES | NxCF_EXCLUDE_QUOTES2, 0, nxCoreCallback);
    return 0;
}
import net.nanex.NxCoreClass;
class CategoryDumper extends NxCoreClass{
    void categoryDump(NxCoreSystem nxCoreSys, NxCoreMessage nxCoreMsg) {
        NxCoreHeader ch = nxCoreMsg.coreHeader;
        NxCoreCategory cat = nxCoreMsg.coreData.Category;
        String symbol;

        if (ch.pnxStringSymbol.String.charAt(0) == 'o' && ch.pnxOptionHdr != null) {// If an option....
            symbol = String.format("%s%02d%02d%02d%c%08d",
            ch.pnxStringSymbol.String,
            ch.pnxOptionHdr.nxExpirationDate.Year-2000,
            ch.pnxOptionHdr.nxExpirationDate.Month,
            ch.pnxOptionHdr.nxExpirationDate.Day,
            (ch.pnxOptionHdr.PutCall == 0) ? 'C' : 'P',
            ch.pnxOptionHdr.strikePrice);
        }
        else{// otherwise a non-option
            symbol = ch.pnxStringSymbol.String;
        }

        System.out.println(String.format("Category         : %s(%d)",          cat.pnxStringCategory.String, cat.pnxStringCategory.Atom));
        System.out.println(String.format("  nxTime         : %02d:%02d:%02d",   (int) nxCoreSys.nxTime.Hour, (int) nxCoreSys.nxTime.Minute, (int) nxCoreSys.nxTime.Second));
        System.out.println(String.format("  symbol         : %s",               symbol));
        System.out.println(String.format("  listedExch     : %d",              ch.ListedExg));
        System.out.println(String.format("  sessionDate    : %d%02d%02d",       (int) ch.nxSessionDate.Year, (int) ch.nxSessionDate.Month, (int) ch.nxSessionDate.Day));
        System.out.println(String.format("  time           : %02d:%02d:%02d",   (int) ch.nxExgTimestamp.Hour, (int) ch.nxExgTimestamp.Minute, (int) ch.nxExgTimestamp.Second));
        System.out.println(String.format("  reportExch     : %d",              ch.ReportingExg));
        System.out.println(String.format("  sessionId      : %d",              (int) ch.SessionID));
        System.out.println(String.format("  permissionId   : %d [%s]",         ch.PermissionID, GetDefinedString(defines.NxST_PERMID, ch.PermissionID)));
        System.out.println(String.format("  NFields        : %d",              cat.NFields));

        for (int i = 0; i < cat.NFields; i++)
        {
            NxCategoryField cf = cat.pnxFields[i];

            System.out.print(String.format("     %d|%s|[%s] {%d} :", i, cf.FieldName, cf.FieldInfo, (int) cf.Set));

            if (cf.Set !=0)
            {
                System.out.println("  -- !Set --");
                continue;
            }

            switch (cf.FieldType)
            {
                case defines.NxCFT_UNKNOWN:
                {
                    System.out.print("  -- unknown --");
                    break;
                }
                case defines.NxCFT_64BIT:
                {
                    System.out.print(String.format("  %d", cf.data.i64Bit));
                    break;
                }
                case defines.NxCFT_32BIT:
                {
                    System.out.print(String.format("  %d", cf.data.i32Bit));
                    break;
                }
                case defines.NxCFT_STRINGZ:
                {
                    System.out.print(String.format("  %s", cf.data.StringZ));
                    break;
                }
                case defines.NxCFT_DOUBLE:
                {
                    System.out.print(String.format("  %f", cf.data.Double));
                    break;
                }
                case defines.NxCFT_PRICE:
                {
                    System.out.print(String.format("  %.2f", PriceToDouble(cf.data.nxPrice.Price, cf.data.nxPrice.PriceType)));
                    break;
                }
                case defines.NxCFT_DATE:
                {
                    System.out.print(String.format("  %d%02d%02d",  (int) cf.data.nxDate.Year, (int) cf.data.nxDate.Month, (int) cf.data.nxDate.Day));
                    break;
                }
                case defines.NxCFT_TIME:
                {
                    System.out.print(String.format("  %02d:%02d:%02d.%03d", (int) cf.data.nxTime.Hour, (int) cf.data.nxTime.Minute, (int) cf.data.nxTime.Second, (int) cf.data.nxTime.Millisecond));
                    break;
                }
                case defines.NxCFT_NxSTRING:
                {
                    if (cf.data.pnxString != null)
                    {
                        NxString s = cf.data.pnxString;
                        System.out.print(String.format("  '%s'", s.String));
                    }
                    break;
                }
                case defines.NxCFT_STRING_IDX:
                {
                    StringTableItem stItem = cf.data.stringTableItem;
                    String ss = GetDefinedString(stItem.ixTable, stItem.idString);
                    if (ss == null)
                    {
                        ss = "***NULL***";
                    }
                    System.out.print(String.format("  [%d:%d] '%s'", stItem.ixTable, stItem.idString, ss));
                    break;
                }
                case defines.NxCFT_STRING_MAP:
                {
                    StringTableItem stItem = cf.data.stringTableItem;
                    boolean printed = false;
                    int id = stItem.idString;
                    for (int iMap = 0; id!=0; ++iMap, id >>= 1)
                    {
                        if ((id & 1) != 0)
                        {
                            String ss = GetDefinedString(stItem.ixTable, iMap);
                            if (ss == null)
                            {
                                ss = "***NULL***";
                            }
                            if (!printed)
                            {
                                System.out.print(String.format("  [%d:%d] '%s'", stItem.ixTable, iMap, ss));
                                printed = true;
                            }
                            else
                            {
                                System.out.print(String.format(",  [%d:%d] '%s'", stItem.ixTable, iMap, ss));
                            }
                        }
                    }
                    break;
                }
            }

            System.out.println("");
        }
    }
    @Override
    public int OnNxCoreCallback(NxCoreSystem nxCoreSys, NxCoreMessage nxCoreMsg) {
        switch (nxCoreMsg.MessageType)
        {
            case defines.NxMSG_EXGQUOTE:

                categoryDump(nxCoreSys,nxCoreMsg);
                break;
        }
        return defines.NxCALLBACKRETURN_CONTINUE;
    }

    public static void main(String args[]) {
        CategoryDumper nxCore = new CategoryDumper();

        if (nxCore.LoadNxCore("NxCoreAPI64.dll") != 0){
            nxCore.ProcessTape(args[0], 0, defines.NxCF_EXCLUDE_CRC_CHECK | defines.NxCF_EXCLUDE_QUOTES | defines.NxCF_EXCLUDE_QUOTES2, 0);
        }
        else
            System.out.println("loading library failed");
    }
}
import NxCore

tapePath = ""
def categoryDump(nxCoreSys, nxCoreMsg):
    ch = nxCoreMsg.coreHeader
    cat = nxCoreMsg.coreData.Category

    if ch.pnxStringSymbol.String[0] == 'o' and ch.pnxOptionHdr: # If an option....
        symbol = "{}{:02d}{:02d}{:02d}{}{:08d}".format(
        ch.pnxStringSymbol.String,
        ch.pnxOptionHdr.nxExpirationDate.Year-2000,
        ch.pnxOptionHdr.nxExpirationDate.Month,
        ch.pnxOptionHdr.nxExpirationDate.Day,
        'C' if ch.pnxOptionHdr.PutCall == 0 else 'P',
        ch.pnxOptionHdr.strikePrice)
    else:# otherwise a non-option
        symbol = ch.pnxStringSymbol.String

    print("Category         : {}({})".format(cat.pnxStringCategory.String, cat.pnxStringCategory.Atom))
    print("  nxTime         : {:02d}:{:02d}:{:02d}".format( nxCoreSys.nxTime.Hour, nxCoreSys.nxTime.Minute, nxCoreSys.nxTime.Second))
    print("  symbol         : {}".format(symbol))
    print("  listedExch     : {}".format(ch.ListedExg))
    print("  sessionDate    : {}{:02d}{:02d}".format(ch.nxSessionDate.Year, ch.nxSessionDate.Month, ch.nxSessionDate.Day))
    print("  time           : {:02d}:{:02d}:{:02d}".format(ch.nxExgTimestamp.Hour, ch.nxExgTimestamp.Minute, ch.nxExgTimestamp.Second))
    print("  reportExch     : {}".format(ch.ReportingExg))
    print("  sessionId      : {}".format(ch.SessionID))
    print("  permissionId   : {} [{}]".format(ch.PermissionID, str(NxCore.GetDefinedString(NxCore.NxST_PERMID, ch.PermissionID))))
    print("  NFields        : {}".format(cat.NFields))

    for i in range(cat.NFields):
        cf = cat.pnxFields[i]

        print("     {}|{}|[{}] {{}} :".format(i, cf.FieldName, cf.FieldInfo, cf.Set), end='')

        if not cf.Set:
            print("  -- !Set --")
            continue

        if cf.FieldType == NxCore.NxCFT_UNKNOWN:
            print("  -- unknown --", end='')
        elif cf.FieldType == NxCore.NxCFT_64BIT:
            print("  {}".format(cf.data.i64Bit), end='')
        elif cf.FieldType == NxCore.NxCFT_32BIT:
            print("  {}".format(cf.data.i32Bit), end='')
        elif cf.FieldType == NxCore.NxCFT_STRINGZ:
            print("  {}".format(cf.data.StringZ), end='')
        elif cf.FieldType == NxCore.NxCFT_DOUBLE:
            print("  {:f}".format(cf.data.Double), end='')
        elif cf.FieldType == NxCore.NxCFT_PRICE:
            print("  {:.2f}".format(NxCore.PriceToFloat(cf.data.nxPrice.Price, cf.data.nxPrice.PriceType)), end='')
        elif cf.FieldType == NxCore.NxCFT_DATE:
            print("  {}{:02d}{:02d}".format(cf.data.nxDate.Year, cf.data.nxDate.Month, cf.data.nxDate.Day), end='')
        elif cf.FieldType == NxCore.NxCFT_TIME:
            print("  {:02d}:{:02d}:{:02d}.{:03d}".format( cf.data.nxTime.Hour, cf.data.nxTime.Minute, cf.data.nxTime.Second, cf.data.nxTime.Millisecond), end='')
        elif cf.FieldType == NxCore.NxCFT_NxSTRING:
            if cf.data.pnxString:
                s = cf.data.pnxString
                print("  '{}'".format(s.String), end='')
        elif cf.FieldType == NxCore.NxCFT_STRING_IDX:
            stItem = cf.data.stringTableItem
            ss = NxCore.GetDefinedString(stItem.ixTable, stItem.idString)
            if not ss:
                ss = "***NULL***"
            print("  [{}:{}] '{}'".format(stItem.ixTable, stItem.idString, ss), end='')
        elif cf.FieldType == NxCore.NxCFT_STRING_MAP:
            stItem = cf.data.stringTableItem
            printed = False
            id = stItem.idString
            iMap = 0
            while id != 0:

                if (id & 1) != 0:
                    ss = NxCore.GetDefinedString(stItem.ixTable, iMap)
                    if not ss :
                        ss = "***NULL***"
                    if not printed:
                        print("  [{}:{}] '{}'".format(stItem.ixTable, iMap, ss), end='')
                        printed = true
                    else:
                        print(",  [{}:{}] '{}'".format(stItem.ixTable, iMap, ss), end='')
                iMap+= 1
                id = id >> 1

        print("")

def OnNxCoreCallback(NxCoreSys, NxCoreMsg):
    if NxCoreMsg.MessageType == NxCore.NxMSG_EXGQUOTE:
        categoryDump(NxCoreSys, NxCoreMsg)
    return NxCore.NxCALLBACKRETURN_CONTINUE

if NxCore.LoadNxCore("NxCoreAPI64.dll"):
NxCore.ProcessTape(tapePath, 0, NxCore.NxCF_EXCLUDE_CRC_CHECK | NxCore.NxCF_EXCLUDE_QUOTES | NxCore.NxCF_EXCLUDE_QUOTES2, 0, OnNxCoreCallback)
else:
    print("loading library failed")