Topics
Replies
netread2004
29 Nov 2018, 18:11
Thanks for your reply.
I am using simple parsing function in OnTick process and that does the parsing of all received messages.
Is there any other way that can do this parsing more frequent than each Tick?
I am now parsing all received messages irrespective of the type of message I send. Is it not same as listener function?
Regarding 35=AF, I do not see this function in FIX API Sample code. I am trying to incorporate this function in my code but I have few questions on Tag 584 and 585. What should be the value of 584? Is it the same as ClOrdID ( the value of 11)? What should be the value of 585? Should it be 1 (Status for Orders for a Secuirty)?
Please let me know.
@netread2004
netread2004
23 Nov 2018, 18:58
Now I get following BUILD error:
Error CS0117: 'cAlgo.API.Chart' does not contain a definition for 'DrawStaticText'
Error CS0103: The name 'Color' does not exist in the current context
@netread2004
netread2004
22 Nov 2018, 17:23
Hi Panagiotis,
I created two seperate counters for Sequence numbers and it is working now. I get execution report after I send Stop Order.
Thanks for your help.
I have more general question on using this FIX API sample code to implement my FIX Engine based Robot -- What I see here is this piece of code from Sample FIX API works perfectly alright and as long as I send correct FIX messages and decrypt received FIX messages correctly and implement logic around this send/receive info I think there should NOT be any other issue in implementing 100% correct Robot. I am not checking CheckSum seperatly to validate the messages but I think it should not matter much. Apart from these concern what do you think there could be other issues that may affect implementation of Robot down the line?
Ypur input on this will be very helpful to move forward in implementing this Robot.
Please let me know.
@netread2004
netread2004
22 Nov 2018, 16:04
RE:
Panagiotis Charalampous said:
Hi netread2004,
I used the FIX API example application to send the order. I would suggest you try it as well and make sure the order is placed. Maybe something was messed up on code transfering.
Best Regards,
Panagiotis
I used same FIX API example application as well but I can see the same issue there. Probably you did not log on to QUOTE server and did not subscribe to SPOT Market Data that is the reason you are able to get the execution report. I get Execution report when I DO NOT log on to Quote server and DO NOT subscribe to SPOT Market Data.
Could you please try using the same sequence as I used so that you can reproduce this issue.
@netread2004
netread2004
22 Nov 2018, 16:01
Hi Panagiotis,
Did you use the same sequence of command I used?
1. logOn to Quote
2. Subscribe to Spot Market Data for EURUSD and GBPUSD
3. LogOn to TRADE
4. Place StopOrder for GBPUSD
Because when I don't subscribe to SPOT Market data I get execution report in response to Stop Order message.
This happens only when I LogOn to QUOTE and Subscribe to SPOT Market Data.
Please try this from your side to reproduce this issue.
@netread2004
netread2004
22 Nov 2018, 15:49
Yes I can see that Order ID 37=31959 (as OID31959 in cTrader) is placed in my cTrader.
Not sure what the issue is with my environment.
Is there anything you can suggest me to try to root-cause this issue in my environment?
@netread2004
netread2004
22 Nov 2018, 13:23
Here is the Stop Order with 59=1 (GTC) sequence:
1)Stop Order 59=1
22/11/2018 11:15:25.995 | SendText : 8=FIX.4.4 9=158 35=D 49=circlemarkets.4000022 56=cServer 57=TRADE 50=TRADE 34=5 52=20181122-11:15:23 11=836669846 55=2 54=1 60=20181122-11:15:23 38=1000 40=3 99=1.28907 59=1 10=186
22/11/2018 11:15:25.995 | ReceivedText : 8=FIX.4.4 9=98 35=2 34=2 49=cServer 50=TRADE 52=20181122-11:15:26.004 56=circlemarkets.4000022 57=TRADE 7=2 16=0 10=071
2)Order Status check 35=H 34=6
22/11/2018 11:15:56.058 | SendText : 8=FIX.4.4 9=103 35=H 49=circlemarkets.4000022 56=cServer 57=TRADE 50=TRADE 34=6 52=20181122-11:15:24 11=836669846 54=1 10=164
22/11/2018 11:15:56.058 | ReceivedText : 8=FIX.4.4 9=89 35=0 34=3 49=cServer 50=TRADE 52=20181122-11:15:56.062 56=circlemarkets.4000022 57=TRADE 10=209
3)Resend Order Status Check 7=6 16=6
22/11/2018 11:15:56.276 | SendText : 8=FIX.4.4 9=94 35=2 49=circlemarkets.4000022 56=cServer 57=TRADE 50=TRADE 34=7 52=20181122-11:15:54 7=6 16=6 10=145
22/11/2018 11:15:56.292 | ReceivedText : 8=FIX.4.4 9=127 35=4 34=6 43=Y 49=cServer 50=TRADE 52=20181122-11:15:56.267 56=circlemarkets.4000022 57=TRADE 122=20181122-11:15:56 36=4 123=Y 10=070
I receive 35=4 for above Rsend Order and it goes on in loop, I send 35=2 and in response I get 35=4
@netread2004
netread2004
22 Nov 2018, 12:07
Here is the sequence of messages right from the begining LogOn to QUOTE server:
1)LogOn to Quote
22/11/2018 09:49:22.168 | SendText : 8=FIX.4.4 9=128 35=A 49=circlemarkets.4000022 56=cServer 57=QUOTE 50=TRADE 34=1 52=20181122-09:49:19 98=0 108=30 141=Y 553=4000022 554=qwerty12 10=211
22/11/2018 09:49:22.168 | ReceivedText : 8=FIX.4.4 9=107 35=A 34=1 49=cServer 50=QUOTE 52=20181122-09:49:22.069 56=circlemarkets.4000022 57=TRADE 98=0 108=30 141=Y 10=121
2)Get Spot Market data for EURUSD
22/11/2018 09:49:22.371 | SendText : 8=FIX.4.4 9=150 35=V 49=circlemarkets.4000022 56=cServer 57=QUOTE 50=TRADE 34=2 52=20181122-09:49:20 262=EURUSD:WDqsoT 263=1 264=1 265=1 267=2 269=0 269=1 146=1 55=1 10=214
22/11/2018 09:49:22.371 | ReceivedText : 8=FIX.4.4 9=136 35=W 34=2 49=cServer 50=QUOTE 52=20181122-09:49:22.276 56=circlemarkets.4000022 57=TRADE 55=1 268=2 269=0 270=1.14064 269=1 270=1.14068 10=194
3)Get Spot Market data for GBPUSD
22/11/2018 09:49:22.574 SendText : 8=FIX.4.4 9=150 35=V 49=circlemarkets.4000022 56=cServer 57=QUOTE 50=TRADE 34=3 52=20181122-09:49:20 262=EURUSD:WDqsoT 263=1 264=1 265=1 267=2 269=0 269=1 146=1 55=2 10=216
22/11/2018 09:49:22.574 | ReceivedText : 8=FIX.4.4 9=136 35=W 34=3 49=cServer 50=QUOTE 52=20181122-09:49:22.484 56=circlemarkets.4000022 57=TRADE 55=2 268=2 269=0 270=1.27906 269=1 270=1.27913 10=209
4)LogOn to TRADE server
22/11/2018 09:49:22.840 | SendText : 8=FIX.4.4 9=122 35=A 49=circlemarkets.4000022 56=cServer 57=TRADE 50=TRADE 34=4 52=20181122-09:49:20 98=0 108=30 553=4000022 554=qwerty12 10=125
22/11/2018 09:49:22.840 | ReceivedText : 8=FIX.4.4 9=101 35=A 34=1 49=cServer 50=TRADE 52=20181122-09:49:22.738 56=circlemarkets.4000022 57=TRADE 98=0 108=30 10=043
5)Send New Stop Order 35=D 40=3, 54=2 for GBPUSD 55=2
22/11/2018 09:49:39.574 | SendText : 8=FIX.4.4 9=158 35=D 49=circlemarkets.4000022 56=cServer 57=TRADE 50=TRADE 34=7 52=20181122-09:49:37 11=976009052 55=2 54=2 60=20181122-09:49:37 38=1000 40=3 99=1.27889 59=3 10=219
22/11/2018 09:49:39.574 | ReceivedText : 8=FIX.4.4 9=98 35=2 34=2 49=cServer 50=TRADE 52=20181122-09:49:32.955 56=circlemarkets.4000022 57=TRADE 7=2 16=0 10=097
6)I send OrderStatus Check 35=H
22/11/2018 09:50:03.153 | SendText : 8=FIX.4.4 9=103 35=H 49=circlemarkets.4000022 56=cServer 57=TRADE 50=TRADE 34=8 52=20181122-09:49:37 11=976009052 54=2 10=167
22/11/2018 09:50:03.153 ReceivedText : 8=FIX.4.4 9=89 35=0 34=3 49=cServer 50=TRADE 52=20181122-09:50:03.063 56=circlemarkets.4000022 57=TRADE 10=208
7)I Send Resend Request for above Order Status Check 7=8 16=8
22/11/2018 09:50:03.356 | SendText : 8=FIX.4.4 9=94 35=2 49=circlemarkets.4000022 56=cServer 57=TRADE 50=TRADE 34=9 52=20181122-09:50:01 7=8 16=8 10=149
22/11/2018 09:50:03.356 | ReceivedText : 8=FIX.4.4 9=127 35=4 34=8 43=Y 49=cServer 50=TRADE 52=20181122-09:50:03.265 56=circlemarkets.4000022 57=TRADE 122=20181122-09:50:03 36=4 123=Y 10=066
8)I receieve ResetSequence message and I keep sending ResendRequest and this goes on in loop and Stop Order is never placed.
22/11/2018 09:50:03.559 | SendText : 8=FIX.4.4 9=95 35=2 49=circlemarkets.4000022 56=cServer 57=TRADE 50=TRADE 34=10 52=20181122-09:50:01 7=9 16=9 10=192
22/11/2018 09:50:03.559 | ReceivedText : 8=FIX.4.4 9=127 35=4 34=9 43=Y 49=cServer 50=TRADE 52=20181122-09:50:03.462 56=circlemarkets.4000022 57=TRADE 122=20181122-09:50:03 36=4 123=Y 10=066
Please let me know if you need anything else.
thanks
@netread2004
netread2004
21 Nov 2018, 18:19
Hi Panagiotis,
Sorry what I meant was my Robot did not work that is it stopped taking STOP orders.
When I added Tag 7 it stopped giving me "Missing Tag" error but my Robot stopped working because it never received any FIX message with 35=8. It always received 35=4 (Sequence Reset) message. So, obviously which does not indicate any STOP Order placed.
Here are the FIX messages sent and received when it strated giving Resend Request to first Order Check Message:
21/11/2018 16:02:23.080 | SendText : 8=FIX.4.4 9=104 35=H 49=circlemarkets.4000022 56=cServer 57=TRADE 50=TRADE 34=6 52=20181121-16:01:51 11=1362478880 54=1 10=203
21/11/2018 16:02:23.080 | ReceivedText : 8=FIX.4.4 9=89 35=0 34=3 49=cServer 50=TRADE 52=20181121-16:02:23.062 56=circlemarkets.4000022 57=TRADE 10=203
21/11/2018 16:02:23.315 | SendText : 8=FIX.4.4 9=94 35=2 49=circlemarkets.4000022 56=cServer 57=TRADE 50=TRADE 34=7 52=20181121-16:02:21 7=6 16=6 10=139
21/11/2018 16:02:23.315 | ReceivedText : 8=FIX.4.4 9=127 35=4 34=6 43=Y 49=cServer 50=TRADE 52=20181121-16:02:23.281 56=circlemarkets.4000022 57=TRADE 122=20181121-16:02:23 36=4 123=Y 10=054
21/11/2018 16:02:23.533 | SendText : 8=FIX.4.4 9=94 35=2 49=circlemarkets.4000022 56=cServer 57=TRADE 50=TRADE 34=8 52=20181121-16:02:21 7=7 16=7 10=142
21/11/2018 16:02:23.533 | ReceivedText : 8=FIX.4.4 9=127 35=4 34=7 43=Y 49=cServer 50=TRADE 52=20181121-16:02:23.526 56=circlemarkets.4000022 57=TRADE 122=20181121-16:02:23 36=4 123=Y 10=057
As you can see the first message from the above is 35=H (Order Check Request) with 34=6 and in response I received 35=0 which I can ignore I guess. Following this message it sent Resend Request (35=2) with 7=6 ( 6 is the sequence number of earlier Order Check message indicating it is Resending this Order Check Message) and since there in only one message to send 7=6 as well as 16=6.
After this Resent request it sends another Resend Request and it receives all the time 35=4 (Sequence Reset) message and it never received 35=8. This goes on in the loop and therefore my Robot wont take any stop order.
Not sure what is the meaning of this received message 35=4. Does it mean I need to send Reset Response message. Not sure. Please let me know.
Thanks for your prompt response to my earlier message.
@netread2004
netread2004
21 Nov 2018, 14:33
Thanks for clarification.
So, there is definitly some advantage in using FIX message for trading.
Thanks again for your help in understanding FIX.
@netread2004
netread2004
19 Nov 2018, 17:52
Hi Panagiotis,
I hate to ask you same question again but please bare with me.
If there is no execution/speed advatage by using QUOTE and the price received via FIX is same as cTrader then is there any other advantage of using QUOTE stream?
If I want to think in terms of speed in communicating with FIX server then this QUOTE thing is redundant is it not as far as reducing slippage?
Is there any reference that gives detailed info about this? I don't see this question very obvious although I am new to this field.
There must be some advantage of using QUOTE stream and I am only interested in using it if I get any advantage in terms of slippage or spread.
From your messages it looks like there is none.
Your comments will be greatly appreciated.
Thanks
@netread2004
netread2004
19 Nov 2018, 13:50
Thanks for your response.
So, what are the advantages of quote price over the normal price?
Is there any SPREAD advatage?
Looks like there could be spread advantage as QUOTE comes from directly LPs and prices without quote comes from cTrader Broker. Am I right?
@netread2004
netread2004
08 Nov 2018, 10:45
I debugged this issue further and found that string "" (empty string) was being passed to Convert.ToDouble("") function was causing this error.
So, this issue is sloved and considered as closed.
Thanks
@netread2004
netread2004
05 Nov 2018, 09:31
Looks like this is logON issue. I checked response for LogOn FIX message and the response is empty message.
As far as my original issue is concerend, I think it was arising due to sending logOn message to the server already Logged on and no Logout sent.
I will monitor this further and update on this.
Thanks
@netread2004
netread2004
05 Nov 2018, 02:34
RE:
Panagiotis Charalampous said:
Hi netread2004,
Can you share the cBot code so that we can reproduce the issue?
Best Regards,
Panagiotis
Here is the error messages I am getting now when I run this code.
05/11/2018 00:27:49.254 | cBot "FIXengine_iCMarkets_test" was started successfully for GBPUSD, m1.
05/11/2018 00:28:09.613 | Crashed in OnTimer with IOException: Unable to write data to the transport connection: An existing connection was forcibly closed by the remote host.
05/11/2018 00:28:09.613 | Crashed in OnStop with IOException: Unable to write data to the transport connection: An existing connection was forcibly closed by the remote host.
05/11/2018 00:28:09.629 | cBot "FIXengine_iCMarkets_test" was stopped for GBPUSD, m1.
Please change FIX credential appropriatly to match with your server while testing it in your environment.
//using FIX_API_Library; using System; using System.Linq; using cAlgo.API; using cAlgo.API.Indicators; using cAlgo.API.Internals; using cAlgo.Indicators; using System.Net.Security; using System.Net.Sockets; using System.Security.Cryptography.X509Certificates; using System.Text; using System.Threading; namespace cAlgo { [Robot(TimeZone = TimeZones.UTC, AccessRights = AccessRights.FullAccess)] public partial class frmFIXAPISample : Robot { [Parameter("Source")] public DataSeries Source { get; set; } [Parameter("Stop Loss (pips)", DefaultValue = 1.3, MinValue = 1)] public double StopLossInPips { get; set; } [Parameter("Take Profit (pips)", DefaultValue = 100.0, MinValue = 1)] public double TakeProfitInPips { get; set; } [Parameter("Quantity (Lots)", DefaultValue = 0.01, MinValue = 0.01, Step = 0.01)] public double Quantity { get; set; } [Parameter("Bollinger Bands Deviations", DefaultValue = 0.5)] public double Deviations { get; set; } [Parameter("Bollinger Bands Periods", DefaultValue = 2)] public int Periods { get; set; } [Parameter("Bollinger Bands MA Type")] public MovingAverageType MAType { get; set; } [Parameter("BarHeight Thresh", DefaultValue = 16)] public int BarHeightThresh { get; set; } [Parameter("pipUpDn Thresh", DefaultValue = 180)] public double pipUpDnThresh { get; set; } [Parameter("Tralling Stop (pips)", DefaultValue = 1.3)] public double StopLoss { get; set; } [Parameter("Inital Stop Loss (pips)", DefaultValue = 1.3)] public double init_StopLoss { get; set; } [Parameter("RiskPC", DefaultValue = 10.0)] public double risk { get; set; } [Parameter("AllowedSpread", DefaultValue = 2.0)] public double allowedSpread { get; set; } [Parameter("toDisp", DefaultValue = true)] public bool toDisp { get; set; } [Parameter("ordNumDisp", DefaultValue = false)] public bool ordNumDisp { get; set; } [Parameter("printWholeFixMsg", DefaultValue = false)] public bool printWholeFixMsg { get; set; } [Parameter("symbolID", DefaultValue = 1)] public int symbolID { get; set; } public const int EURUSD = 1; public const int GBPUSD = 2; string txtMessageSend_Text = ""; string txtMessageReceived_Text = ""; bool deCryptEn = true; bool deCryptNoPrint = false; bool stopLossPlaced = false; bool lookForPrice = false; bool filledDueToSLHit = false; string initPriceForSL; string randomNum = ""; string[] filledPosiitonID = { "" }; string filPosID; string avePricePosition; string clOrdNumGlob; bool origStopOrdFilled = false; int totalPosGlob = 0; //Ctrader private int _pricePort = 5211; private int _tradePort = 5212; private string _host = "h1.p.ctrader.cn"; private string _username = "3388140"; private string _password = "qwerty12"; private string _senderCompID = "icmarkets.3388140"; private string _targetCompID = "cServer"; private string _senderSubID = "TRADE"; private int _messageSequenceNumber = 1; private int _testRequestID = 1; private TcpClient _priceClient; private SslStream _priceStreamSSL; private TcpClient _tradeClient; private SslStream _tradeStreamSSL; private MessageConstructor _messageConstructor; private double High; private double Low; double spread; string globIDForCheckOrder = ""; string globIdForSLCheckStat = ""; protected override void OnStart() { btnLogonT_Click(); Timer.TimerTick += OnTimerTick; Timer.Start(10); Positions.Closed += PositionsOnClosed; } protected override void OnStop() { btnLogoutT_Click(); } private void OnTimerTick() { btnHeartbeatT_Click(); } private void PositionsOnClosed(PositionClosedEventArgs args) { var position = args.Position; Print("Position closed with {0} profit", position.GrossProfit); filledDueToSLHit = false; stopLossPlaced = false; clOrdNumGlob = ""; origStopOrdFilled = false; initPriceForSL = ""; Print("Order Closed Successfully!!!! -------------"); filPosID = ""; globIDForCheckOrder = ""; globIdForSLCheckStat = ""; } protected override void OnTick() { High = MarketSeries.High.Last(0); Low = MarketSeries.Low.Last(0); int totalPositions = Positions.Count; var totalPendOrders = PendingOrders.Count; var barHeight = High - Low; totalPosGlob = totalPositions; if (origStopOrdFilled && totalPositions == 0) { } spread = (Symbol.Ask - Symbol.Bid) / Symbol.PipSize; } private string getRandomNumber(string coment) { Random random = new Random(); int randomNumber = random.Next(); randomNum = randomNumber.ToString(); if (ordNumDisp) Print(coment + " ClOrdeID : ", randomNum); return (randomNum); } private void printSendReceive() { Print("SendText : ", txtMessageSend_Text.Replace("\u0001", " ")); Print("ReceivedText : ", txtMessageReceived_Text.Replace("\u0001", " ")); } private void deCryptMessage(string sentMssg, string recvdMessg) { } //End of my routines public frmFIXAPISample() { //InitializeComponent(); _priceClient = new TcpClient(_host, _pricePort); _priceStreamSSL = new SslStream(_priceClient.GetStream(), false, new RemoteCertificateValidationCallback(ValidateServerCertificate), null); _priceStreamSSL.AuthenticateAsClient(_host); _tradeClient = new TcpClient(_host, _tradePort); _tradeStreamSSL = new SslStream(_tradeClient.GetStream(), false, new RemoteCertificateValidationCallback(ValidateServerCertificate), null); _tradeStreamSSL.AuthenticateAsClient(_host); _messageConstructor = new MessageConstructor(_host, _username, _password, _senderCompID, _senderSubID, _targetCompID); } private static bool ValidateServerCertificate(object sender, X509Certificate certificate, X509Chain chain, SslPolicyErrors sslPolicyErrors) { if (sslPolicyErrors == SslPolicyErrors.None) return true; Console.WriteLine("Certificate error: {0}", sslPolicyErrors); return false; } private void ClearText() { txtMessageSend_Text = ""; txtMessageReceived_Text = ""; } private string SendPriceMessage(string message, bool readResponse = true) { return SendMessage(message, _priceStreamSSL, readResponse); } private string SendTradeMessage(string message, bool readResponse = true) { return SendMessage(message, _tradeStreamSSL, readResponse); } private string SendMessage(string message, SslStream stream, bool readResponse = true) { var byteArray = Encoding.ASCII.GetBytes(message); System.Net.ServicePointManager.Expect100Continue = false; //I inserted Above line to see if it fixes crash issue stream.Write(byteArray, 0, byteArray.Length); var buffer = new byte[1024]; if (readResponse) { Thread.Sleep(100); stream.Read(buffer, 0, 1024); } _messageSequenceNumber++; var returnMessage = Encoding.ASCII.GetString(buffer); return returnMessage; } private void btnLogon_Click() { ClearText(); var message = _messageConstructor.LogonMessage(MessageConstructor.SessionQualifier.QUOTE, _messageSequenceNumber, 30, false); txtMessageSend_Text = message; txtMessageReceived_Text = SendPriceMessage(message); if (printWholeFixMsg) printSendReceive(); } private void btnTestRequest_Click(object sender, EventArgs e) { ClearText(); var message = _messageConstructor.TestRequestMessage(MessageConstructor.SessionQualifier.QUOTE, _messageSequenceNumber, _testRequestID); _testRequestID++; txtMessageSend_Text = message; txtMessageReceived_Text = SendPriceMessage(message); } private void btnLogout_Click() { ClearText(); var message = _messageConstructor.LogoutMessage(MessageConstructor.SessionQualifier.QUOTE, _messageSequenceNumber); txtMessageSend_Text = message; txtMessageReceived_Text = SendPriceMessage(message); _messageSequenceNumber = 1; if (printWholeFixMsg) printSendReceive(); } private void btnMarketDataRequest_Click(object sender, EventArgs e) { ClearText(); var message = _messageConstructor.MarketDataRequestMessage(MessageConstructor.SessionQualifier.QUOTE, _messageSequenceNumber, "EURUSD:WDqsoT", 1, 0, 0, 1, 1); txtMessageSend_Text = message; txtMessageReceived_Text = SendPriceMessage(message); } private void btnHeartbeat_Click() { ClearText(); var message = _messageConstructor.HeartbeatMessage(MessageConstructor.SessionQualifier.QUOTE, _messageSequenceNumber); txtMessageSend_Text = message; txtMessageReceived_Text = SendPriceMessage(message, false); if (printWholeFixMsg) printSendReceive(); } private void btnResendRequest_Click(object sender, EventArgs e) { ClearText(); var message = _messageConstructor.ResendMessage(MessageConstructor.SessionQualifier.QUOTE, _messageSequenceNumber, _messageSequenceNumber - 1); _testRequestID++; txtMessageSend_Text = message; txtMessageReceived_Text = SendPriceMessage(message); } private void btnReject_Click(object sender, EventArgs e) { ClearText(); var message = _messageConstructor.RejectMessage(MessageConstructor.SessionQualifier.QUOTE, _messageSequenceNumber, 0); _testRequestID++; txtMessageSend_Text = message; txtMessageReceived_Text = SendPriceMessage(message); } private void btnSequenceReset_Click(object sender, EventArgs e) { ClearText(); var message = _messageConstructor.SequenceResetMessage(MessageConstructor.SessionQualifier.QUOTE, _messageSequenceNumber, 0); _testRequestID++; txtMessageSend_Text = message; txtMessageReceived_Text = SendPriceMessage(message); } private void btnNewOrderSingle_Click(object sender, EventArgs e) { ClearText(); var message = _messageConstructor.NewOrderSingleMessage(MessageConstructor.SessionQualifier.TRADE, _messageSequenceNumber, "1408471", 1, 1, DateTime.UtcNow.ToString("yyyyMMdd-HH:mm:ss"), 1000, 1, "1"); _testRequestID++; txtMessageSend_Text = message; txtMessageReceived_Text = SendTradeMessage(message); } private void btnOrderStatusRequest_Click(string orderId, int symbl, int ordtyp) { ClearText(); var message = _messageConstructor.OrderStatusRequest(MessageConstructor.SessionQualifier.TRADE, _messageSequenceNumber, orderId, symbl, ordtyp); _testRequestID++; txtMessageSend_Text = message; txtMessageReceived_Text = SendTradeMessage(message); if (printWholeFixMsg) printSendReceive(); if (deCryptEn) deCryptMessage(txtMessageSend_Text, txtMessageReceived_Text); } private string Timestamp() { return (DateTime.UtcNow.Subtract(new DateTime(1970, 1, 1))).TotalSeconds.ToString(); } private void btnRequestForPositions_Click(string posID) { ClearText(); var message = _messageConstructor.RequestForPositions(MessageConstructor.SessionQualifier.TRADE, _messageSequenceNumber, posID); _testRequestID++; txtMessageSend_Text = message; txtMessageReceived_Text = SendTradeMessage(message); if (printWholeFixMsg) printSendReceive(); if (deCryptEn) deCryptMessage(txtMessageSend_Text, txtMessageReceived_Text); } private void btnLogonT_Click() { ClearText(); var message = _messageConstructor.LogonMessage(MessageConstructor.SessionQualifier.TRADE, _messageSequenceNumber, 30, false); txtMessageSend_Text = message; txtMessageReceived_Text = SendTradeMessage(message); if (printWholeFixMsg) printSendReceive(); if (deCryptEn) deCryptMessage(txtMessageSend_Text, txtMessageReceived_Text); } private void btnHeartbeatT_Click() { ClearText(); var message = _messageConstructor.HeartbeatMessage(MessageConstructor.SessionQualifier.TRADE, _messageSequenceNumber); txtMessageSend_Text = message; txtMessageReceived_Text = SendTradeMessage(message, false); if (printWholeFixMsg) printSendReceive(); } private void btnTestRequestT_Click(object sender, EventArgs e) { ClearText(); var message = _messageConstructor.TestRequestMessage(MessageConstructor.SessionQualifier.TRADE, _messageSequenceNumber, _testRequestID); _testRequestID++; txtMessageSend_Text = message; txtMessageReceived_Text = SendTradeMessage(message); } private void btnLogoutT_Click() { ClearText(); var message = _messageConstructor.LogoutMessage(MessageConstructor.SessionQualifier.TRADE, _messageSequenceNumber); txtMessageSend_Text = message; txtMessageReceived_Text = SendTradeMessage(message); _messageSequenceNumber = 1; if (printWholeFixMsg) printSendReceive(); if (deCryptEn) deCryptMessage(txtMessageSend_Text, txtMessageReceived_Text); } private void btnResendRequestT_Click(object sender, EventArgs e) { ClearText(); var message = _messageConstructor.ResendMessage(MessageConstructor.SessionQualifier.TRADE, _messageSequenceNumber, _messageSequenceNumber - 1); _testRequestID++; txtMessageSend_Text = message; txtMessageReceived_Text = SendTradeMessage(message); } private void btnSpotMarketData_Click(object sender, EventArgs e) { ClearText(); var message = _messageConstructor.MarketDataRequestMessage(MessageConstructor.SessionQualifier.QUOTE, _messageSequenceNumber, "EURUSD:WDqsoT", 1, 1, 0, 1, 1); txtMessageSend_Text = message; txtMessageReceived_Text = SendPriceMessage(message); } private void btnStopOrder_Click(int symb, int vol, double stpPrice, string ClOrdID, int side, string posiID) { ClearText(); var message = _messageConstructor.NewOrderSingleMessage(MessageConstructor.SessionQualifier.TRADE, _messageSequenceNumber, ClOrdID, symb, side, DateTime.UtcNow.ToString("yyyyMMdd-HH:mm:ss"), vol, 3, "3", 0, (decimal)stpPrice, "", posiID); _testRequestID++; txtMessageSend_Text = message; txtMessageReceived_Text = SendTradeMessage(message); if (printWholeFixMsg) printSendReceive(); if (deCryptEn) deCryptMessage(txtMessageSend_Text, txtMessageReceived_Text); } private void btnCancelOrder_Click(string ClOrdID, string oldOrdID) { ClearText(); var message = _messageConstructor.CancelOrderSingleMessage(MessageConstructor.SessionQualifier.TRADE, _messageSequenceNumber, ClOrdID, oldOrdID, 1, DateTime.UtcNow.ToString("yyyyMMdd-HH:mm:ss"), 3, "3", 0); _testRequestID++; txtMessageSend_Text = message; txtMessageReceived_Text = SendTradeMessage(message); if (printWholeFixMsg) printSendReceive(); if (deCryptEn) deCryptMessage(txtMessageSend_Text, txtMessageReceived_Text); } private void btnLimitOrder_Click() { ClearText(); var message = _messageConstructor.NewOrderSingleMessage(MessageConstructor.SessionQualifier.TRADE, _messageSequenceNumber, "10", 1, 1, DateTime.UtcNow.ToString("yyyyMMdd-HH:mm:ss"), 1000, 2, "3", (decimal)1.08); _testRequestID++; txtMessageSend_Text = message; txtMessageReceived_Text = SendTradeMessage(message); if (printWholeFixMsg) printSendReceive(); } } public class MessageConstructor { public enum SessionMessageType { Logon, Logout, Heartbeat, TestRequest, Resend, Reject, SequenceReset } public enum ApplicationMessageType { MarketDataRequest, MarketDataIncrementalRefresh, NewOrderSingle, CancelOrderSingle, OrderStatusRequest, ExecutionReport, BusinessMessageReject, RequestForPosition, PositionReport } public enum SessionQualifier { QUOTE, TRADE } private string _host; private string _username; private string _password; private string _senderCompID; private string _senderSubID; private string _targetCompID; public MessageConstructor(string host, string username, string password, string senderCompID, string senderSubID, string targetCompID) { _host = host; _username = username; _password = password; _senderCompID = senderCompID; _senderSubID = senderSubID; _targetCompID = targetCompID; } /// <summary> /// Logons the message. /// </summary> /// <param name="qualifier">The session qualifier.</param> /// <param name="messageSequenceNumber">The message sequence number.</param> /// <param name="heartBeatSeconds">The heart beat seconds.</param> /// <param name="resetSeqNum">All sides of FIX session should have sequence numbers reset. Valid value ///is "Y" = Yes(reset)..</param> /// <returns></returns> public string LogonMessage(SessionQualifier qualifier, int messageSequenceNumber, int heartBeatSeconds, bool resetSeqNum) { var body = new StringBuilder(); //Defines a message encryption scheme.Currently, only transportlevel security //is supported.Valid value is "0"(zero) = NONE_OTHER (encryption is not used). body.Append("98=0|"); //Heartbeat interval in seconds. //Value is set in the 'config.properties' file (client side) as 'SERVER.POLLING.INTERVAL'. //30 seconds is default interval value. If HeartBtInt is set to 0, no heart beat message //is required. body.Append("108=" + heartBeatSeconds + "|"); // All sides of FIX session should have //sequence numbers reset. Valid value //is "Y" = Yes(reset). if (resetSeqNum) body.Append("141=Y|"); //The numeric User ID. User is linked to SenderCompID (#49) value (the //user’s organization). body.Append("553=" + _username + "|"); //USer Password body.Append("554=" + _password + "|"); var header = ConstructHeader(qualifier, SessionMessageCode(SessionMessageType.Logon), messageSequenceNumber, body.ToString()); var headerAndBody = header + body; var trailer = ConstructTrailer(headerAndBody); var headerAndMessageAndTrailer = header + body + trailer; return headerAndMessageAndTrailer.Replace("|", "\u0001"); } public string HeartbeatMessage(SessionQualifier qualifier, int messageSequenceNumber) { var header = ConstructHeader(qualifier, SessionMessageCode(SessionMessageType.Heartbeat), messageSequenceNumber, string.Empty); var trailer = ConstructTrailer(header); var headerAndMessageAndTrailer = header + trailer; return headerAndMessageAndTrailer.Replace("|", "\u0001"); } public string TestRequestMessage(SessionQualifier qualifier, int messageSequenceNumber, int testRequestID) { var body = new StringBuilder(); //Heartbeat message ID. TestReqID should be incremental. body.Append("112=" + testRequestID + "|"); var header = ConstructHeader(qualifier, SessionMessageCode(SessionMessageType.TestRequest), messageSequenceNumber, body.ToString()); var headerAndBody = header + body; var trailer = ConstructTrailer(headerAndBody); var headerAndMessageAndTrailer = header + body + trailer; return headerAndMessageAndTrailer.Replace("|", "\u0001"); } public string LogoutMessage(SessionQualifier qualifier, int messageSequenceNumber) { var header = ConstructHeader(qualifier, SessionMessageCode(SessionMessageType.Logout), messageSequenceNumber, string.Empty); var trailer = ConstructTrailer(header); var headerAndMessageAndTrailer = header + trailer; return headerAndMessageAndTrailer.Replace("|", "\u0001"); } /// <summary> /// Resends the message. /// </summary> /// <param name="qualifier">The session qualifier.</param> /// <param name="messageSequenceNumber">Message sequence number of last record in range to be resent.</param> /// <param name="endSequenceNo">The end sequence no.</param> /// <returns></returns> public string ResendMessage(SessionQualifier qualifier, int messageSequenceNumber, int endSequenceNo) { var body = new StringBuilder(); //Message sequence number of last record in range to be resent. body.Append("16=" + endSequenceNo + "|"); var header = ConstructHeader(qualifier, SessionMessageCode(SessionMessageType.Resend), messageSequenceNumber, body.ToString()); var headerAndBody = header + body; var trailer = ConstructTrailer(headerAndBody); var headerAndMessageAndTrailer = header + body + trailer; return headerAndMessageAndTrailer.Replace("|", "\u0001"); } public string RejectMessage(SessionQualifier qualifier, int messageSequenceNumber, int rejectSequenceNumber) { var body = new StringBuilder(); //Referenced message sequence number. body.Append("45=" + rejectSequenceNumber + "|"); var header = ConstructHeader(qualifier, SessionMessageCode(SessionMessageType.Reject), messageSequenceNumber, string.Empty); var headerAndBody = header + body; var trailer = ConstructTrailer(headerAndBody); var headerAndMessageAndTrailer = header + body + trailer; return headerAndMessageAndTrailer.Replace("|", "\u0001"); } public string SequenceResetMessage(SessionQualifier qualifier, int messageSequenceNumber, int rejectSequenceNumber) { var body = new StringBuilder(); //New Sequence Number body.Append("36=" + rejectSequenceNumber + "|"); var header = ConstructHeader(qualifier, SessionMessageCode(SessionMessageType.SequenceReset), messageSequenceNumber, string.Empty); var headerAndBody = header + body; var trailer = ConstructTrailer(headerAndBody); var headerAndMessageAndTrailer = header + body + trailer; return headerAndMessageAndTrailer.Replace("|", "\u0001"); } /// <summary> /// Constructs a message for requesting market data. /// </summary> /// <param name="qualifier">The session qualifier.</param> /// <param name="messageSequenceNumber">The message sequence number.</param> /// <param name="marketDataRequestID">Unique quote request id. New ID for a new subscription, same one as previously used for subscription removal.</param> /// <param name="subscriptionRequestType">1 = Snapshot plus updates (subscribe) 2 = Disable previous snapshot plus update request (unsubscribe).</param> /// <param name="marketDepth">Full book will be provided, 0 = Depth subscription, 1 = Spot subscription.</param> /// <param name="marketDataEntryType">Always set to 2 (both bid and ask will be sent).</param> /// <param name="noRelatedSymbol">Number of symbols requested.</param> /// <param name="symbol">Instrument identificators are provided by Spotware.</param> /// <returns></returns> public string MarketDataRequestMessage(SessionQualifier qualifier, int messageSequenceNumber, string marketDataRequestID, int subscriptionRequestType, int marketDepth, int marketDataEntryType, int noRelatedSymbol, long symbol = 0) { var body = new StringBuilder(); //Unique quote request id. New ID for a new subscription, same one as previously used for subscription removal. body.Append("262=" + marketDataRequestID + "|"); //1 = Snapshot plus updates (subscribe) 2 = Disable previous snapshot plus update request (unsubscribe) body.Append("263=" + subscriptionRequestType + "|"); //Full book will be provided, 0 = Depth subscription, 1 = Spot subscription body.Append("264=" + marketDepth + "|"); //Only Incremental refresh is supported. body.Append("265=1|"); //Always set to 2 (both bid and ask will be sent). body.Append("267=2|"); //Always set to 2 (both bid and ask will be sent). body.Append("269=0|269=1|"); //Number of symbols requested. body.Append("146=" + noRelatedSymbol + "|"); //Instrument identificators are provided by Spotware. body.Append("55=" + symbol + "|"); var header = ConstructHeader(qualifier, ApplicationMessageCode(ApplicationMessageType.MarketDataRequest), messageSequenceNumber, body.ToString()); var headerAndBody = header + body; var trailer = ConstructTrailer(headerAndBody); var headerAndMessageAndTrailer = header + body + trailer; return headerAndMessageAndTrailer.Replace("|", "\u0001"); } /// <summary> /// Constructs a message for requesting market data snapshot. /// </summary> /// <param name="qualifier">The session qualifier.</param> /// <param name="messageSequenceNumber">The message sequence number.</param> /// <param name="symbol">Instrument identificators are provided by Spotware.</param> /// <param name="noMarketDataEntries">Number of entries following.</param> /// <param name="entryType">Valid values are: 0 = BID, 1 = OFFER.</param> /// <param name="entryPrice">Price of Market Data Entry.</param> /// <param name="marketDataRequestID">Unique quote request id. New ID for a new subscription, same one as previously used for subscription removal.</param> /// <returns></returns> public string MarketDataSnapshotMessage(SessionQualifier qualifier, int messageSequenceNumber, long symbol, string noMarketDataEntries, int entryType, decimal entryPrice, string marketDataRequestID = "") { var body = new StringBuilder(); //Unique quote request id. New ID for a new subscription, same one as previously used for subscription removal. body.Append("262=" + marketDataRequestID + "|"); //Instrument identificators are provided by Spotware. body.Append("55=" + symbol + "|"); //Number of entries following. body.Append("268=" + noMarketDataEntries + "|"); //Valid values are: 0 = BID, 1 = OFFER. body.Append("269=" + entryType + "|"); //Price of Market Data Entry. body.Append("270=" + entryPrice + "|"); var header = ConstructHeader(qualifier, ApplicationMessageCode(ApplicationMessageType.MarketDataRequest), messageSequenceNumber, body.ToString()); var headerAndBody = header + body; var trailer = ConstructTrailer(headerAndBody); var headerAndMessageAndTrailer = header + body + trailer; return headerAndMessageAndTrailer.Replace("|", "\u0001"); } /// <summary> /// Markets the data incremental refresh message. /// </summary> /// <param name="qualifier">The session qualifier.</param> /// <param name="messageSequenceNumber">The message sequence number.</param> /// <param name="marketDataRequestID">Unique quote request id. New ID for a new subscription, same one as previously used for subscription removal.</param> /// <param name="noMarketDataEntries">Number of entries following. This repeating group contains a list of all types of Market Data Entries the requester wants to receive.</param> /// <param name="marketDataUpdateAction">Type of Market Data update action. Valid values: 0 = NEW, 2 = DELETE.</param> /// <param name="marketDataEntryType">Valid values are: 0 = BID, 1 = OFFER.</param> /// <param name="marketDataEntryID">ID of Market Data Entry.</param> /// <param name="noRelatedSymbol">The no related symbol.</param> /// <param name="marketDataEntryPrice">Conditionally required when MDUpdateAction <279> = New(0).</param> /// <param name="marketDataEntrySize">Conditionally required when MDUpdateAction <279> = New(0).</param> /// <param name="symbol">/Instrument identificators are provided by Spotware.</param> /// <returns></returns> public string MarketDataIncrementalRefreshMessage(SessionQualifier qualifier, int messageSequenceNumber, string marketDataRequestID, int noMarketDataEntries, int marketDataUpdateAction, int marketDataEntryType, string marketDataEntryID, int noRelatedSymbol, decimal marketDataEntryPrice = 0, double marketDataEntrySize = 0, long symbol = 0) { var body = new StringBuilder(); //Unique quote request id. New ID for a new subscription, same one as previously used for subscription removal. body.Append("262=" + marketDataRequestID + "|"); //Number of entries following. This repeating group contains a list of all types of Market Data Entries the requester wants to receive body.Append("268=" + noMarketDataEntries + "|"); //Type of Market Data update action. Valid values: 0 = NEW, 2 = DELETE body.Append("279=" + marketDataUpdateAction + "|"); //Valid values are: 0 = BID, 1 = OFFER body.Append("269=" + marketDataEntryType + "|"); //ID of Market Data Entry. body.Append("278=" + marketDataEntryID + "|"); //Instrument identificators are provided by Spotware body.Append("55=" + symbol + "|"); if (marketDataEntryPrice > 0) { //Conditionally required when MDUpdateAction <279> = New(0) body.Append("270=" + marketDataEntryPrice + "|"); } if (marketDataEntrySize > 0) { //Conditionally required when MDUpdateAction <279> = New(0) body.Append("271=" + marketDataEntrySize + "|"); } var header = ConstructHeader(qualifier, ApplicationMessageCode(ApplicationMessageType.MarketDataIncrementalRefresh), messageSequenceNumber, string.Empty); var headerAndBody = header + body; var trailer = ConstructTrailer(headerAndBody); var headerAndMessageAndTrailer = header + body + trailer; return headerAndMessageAndTrailer.Replace("|", "\u0001"); //return headerAndMessageAndTrailer; } /// <summary> /// News the order single message. /// </summary> /// <param name="qualifier">The session qualifier.</param> /// <param name="messageSequenceNumber">The message sequence number.</param> /// <param name="orderID">Unique identifier for the order, allocated by the client.</param> /// <param name="symbol">The symbol.</param> /// <param name="side">1= Buy, 2 = Sell.</param> /// <param name="transactTime">Client generated request time.</param> /// <param name="orderQuantity">The fixed currency amount.</param> /// <param name="orderType">1 = Market, the Order will be processed by 'Immediate Or Cancel'scheme(see TimeInForce(59): IOC); /// 2 = Limit, the Order will be processed by 'Good Till Cancel' scheme(see TimeInForce(59): GTC).</param> /// <param name="timeInForce">1 = Good Till Cancel (GTC), it will be active only for Limit Orders (see OrdType(40)) ; /// 3 = Immediate Or Cancel (IOC), it will be active only for Market Orders(see OrdType(40)); /// 6 = Good Till Date(GTD), it will be active only if ExpireTime is defined (see ExpireTime(126)). /// GTD has a high priority, so if ExpireTime is defined, GTD will be used for the Order processing.</param> /// <param name="price">The worst client price that the client will accept. /// Required when OrdType = 2, in which case the order will notfill unless this price can be met.</param> /// /// <param name="stopPrice">Reserved for future use</param> /// <param name="expireTime"> Expire Time in YYYYMMDDHH:MM:SS format.If is assigned then the Order will be processed by 'Good Till Date' scheme /// (see TimeInForce: GTD).</param> /// <param name="positionID">Position ID, where this order should be placed. If not set, new position will be created, it’s id will be returned in ExecutionReport(8) message.</param> /// <returns></returns> public string NewOrderSingleMessage(SessionQualifier qualifier, int messageSequenceNumber, string orderID, long symbol, int side, string transactTime, int orderQuantity, int orderType, string timeInForce, decimal price = 0, decimal stopPrice = 0, string expireTime = "", string positionID = "") { var body = new StringBuilder(); //Unique identifier for the order, allocated by the client. body.Append("11=" + orderID + "|"); //Instrument identificators are provided by Spotware. body.Append("55=" + symbol + "|"); //1= Buy, 2 = Sell body.Append("54=" + side + "|"); // Client generated request time. body.Append("60=" + transactTime + "|"); //The fixed currency amount. body.Append("38=" + orderQuantity + "|"); //1 = Market, the Order will be processed by 'Immediate Or Cancel'scheme(see TimeInForce(59): IOC); //2 = Limit, the Order will be processed by 'Good Till Cancel' scheme(see TimeInForce(59): GTC). body.Append("40=" + orderType + "|"); if (price != 0) { //Reserved for future use. body.Append("44=" + price + "|"); } if (stopPrice != 0) { //The worst client price that the client will accept. //Required when OrdType = 2, in which case the order will notfill unless this price can be met. body.Append("99=" + stopPrice + "|"); } // 1 = Good Till Cancel (GTC), it will be active only for Limit Orders (see OrdType(40)) ; // 3 = Immediate Or Cancel (IOC), it will be active only for Market Orders(see OrdType(40)); // 6 = Good Till Date(GTD), it will be active only if ExpireTime is defined (see ExpireTime(126)). // GTD has a high priority, so if ExpireTime is defined, GTD will be used for the Order processing. body.Append("59=" + timeInForce + "|"); if (expireTime != string.Empty) { // Expire Time in YYYYMMDDHH:MM:SS format.If is assigned then the Order will be processed by 'Good Till Date' scheme // (see TimeInForce: GTD). body.Append("126=" + expireTime + "|"); } if (positionID != string.Empty) { // Position ID, where this order should be placed. If not set, new position will be created, it’s id will be returned in ExecutionReport(8) message. body.Append("721=" + positionID + "|"); } var header = ConstructHeader(qualifier, ApplicationMessageCode(ApplicationMessageType.NewOrderSingle), messageSequenceNumber, body.ToString()); var headerAndBody = header + body; var trailer = ConstructTrailer(headerAndBody); var headerAndMessageAndTrailer = header + body + trailer; return headerAndMessageAndTrailer.Replace("|", "\u0001"); } public string CancelOrderSingleMessage(SessionQualifier qualifier, int messageSequenceNumber, string orderID, string oldOrdID, int side, string transactTime, int orderType, string timeInForce, decimal stopPrice = 0, string expireTime = "", string positionID = "") { var body = new StringBuilder(); //OldID identifier for the order, allocated by the client. body.Append("41=" + oldOrdID + "|"); //Unique identifier for the order, allocated by the client. body.Append("11=" + orderID + "|"); /* following is NOT needed for cancel orderModify Order. body.Append("55=" + symbol + "|"); body.Append("54=" + side + "|"); body.Append("60=" + transactTime + "|"); body.Append("38=" + orderQuantity + "|"); body.Append("40=" + orderType + "|"); body.Append("99=" + price + "|"); //Instrument identificators are provided by Spotware. body.Append("55=" + symbol + "|"); //1= Buy, 2 = Sell body.Append("54=" + side + "|"); // Client generated request time. body.Append("60=" + transactTime + "|"); //The fixed currency amount. body.Append("38=" + orderQuantity + "|"); //1 = Market, the Order will be processed by 'Immediate Or Cancel'scheme(see TimeInForce(59): IOC); //2 = Limit, the Order will be processed by 'Good Till Cancel' scheme(see TimeInForce(59): GTC). body.Append("40=" + orderType + "|"); if (price != 0) { //Reserved for future use. body.Append("44=" + price + "|"); } if (stopPrice != 0) { //The worst client price that the client will accept. //Required when OrdType = 2, in which case the order will notfill unless this price can be met. // body.Append("99=" + stopPrice + "|"); } // 1 = Good Till Cancel (GTC), it will be active only for Limit Orders (see OrdType(40)) ; // 3 = Immediate Or Cancel (IOC), it will be active only for Market Orders(see OrdType(40)); // 6 = Good Till Date(GTD), it will be active only if ExpireTime is defined (see ExpireTime(126)). // GTD has a high priority, so if ExpireTime is defined, GTD will be used for the Order processing. // body.Append("59=" + timeInForce + "|"); //not needed for modify if (expireTime != string.Empty) { // Expire Time in YYYYMMDDHH:MM:SS format.If is assigned then the Order will be processed by 'Good Till Date' scheme // (see TimeInForce: GTD). body.Append("126=" + expireTime + "|"); } if (positionID != string.Empty) { // Position ID, where this order should be placed. If not set, new position will be created, it’s id will be returned in ExecutionReport(8) message. body.Append("721=" + positionID + "|"); } */ var header = ConstructHeader(qualifier, ApplicationMessageCode(ApplicationMessageType.CancelOrderSingle), messageSequenceNumber, body.ToString()); var headerAndBody = header + body; var trailer = ConstructTrailer(headerAndBody); var headerAndMessageAndTrailer = header + body + trailer; return headerAndMessageAndTrailer.Replace("|", "\u0001"); } /// <summary> /// Orders the status request. /// </summary> /// <param name="qualifier">The session qualifier.</param> /// <param name="messageSequenceNumber">The message sequence number.</param> /// <param name="orderID">Unique identifier for the order, allocated by the client.</param> /// <returns></returns> public string OrderStatusRequest(SessionQualifier qualifier, int messageSequenceNumber, string orderID, int symb, int orderTyp) { var body = new StringBuilder(); // Unique identifier for the order, allocated by the client. body.Append("11=" + orderID + "|"); //Symbol - added by me //body.Append("55=" + symb + "|");//it was giving me error // 1 = Buy; 2 = Sell. There is for the FIX compatibility only, so it will be ignored. body.Append("54=" + orderTyp + "|"); var header = ConstructHeader(qualifier, ApplicationMessageCode(ApplicationMessageType.OrderStatusRequest), messageSequenceNumber, body.ToString()); var headerAndBody = header + body; var trailer = ConstructTrailer(headerAndBody); var headerAndMessageAndTrailer = header + body + trailer; return headerAndMessageAndTrailer.Replace("|", "\u0001"); } /// <summary> /// Executions the report. /// </summary> /// <param name="qualifier">The session qualifier.</param> /// <param name="messageSequenceNumber">The message sequence number.</param> /// <param name="cTraderOrderID">cTrader order id..</param> /// <param name="orderStatus"> 0 = New; 1 = Partially filled; 2 = Filled; 8 = Rejected; 4 = Cancelled(When an order is partially filled, "Cancelled" is /// returned signifying Tag 151: LeavesQty is cancelled and will not be subsequently filled); C = Expired.</param> /// <param name="transactTime">Time the transaction represented by this ExecutionReport occurred message(in UTC).</param> /// <param name="symbol">Instrument identificators are provided by Spotware..</param> /// <param name="side">1 = Buy; 2 = Sell.</param> /// <param name="averagePrice">The price at which the deal was filled.For an IOC or GTD order, this is the VWAP(Volume Weighted Average Price) of the filled order.</param> /// <param name="orderQuantity"> The fixed currency amount.</param> /// <param name="leavesQuantity">The amount of the order still to be filled.This is a value between 0 (fully filled) and OrderQty (partially filled).</param> /// <param name="cumQuantity">The total amount of the order which has been filled.</param> /// <param name="orderID"> Unique identifier for the order, allocated by the client.</param> /// <param name="orderType"> 1 = Market; 2 = Limit.</param> /// <param name="price">If supplied in the NewOrderSingle, it is echoed back in this ExecutionReport.</param> /// <param name="stopPrice">If supplied in the NewOrderSingle, it is echoed back in this ExecutionReport.</param> /// <param name="timeInForce">U1 = Good Till Cancel (GTC); 3 = Immediate Or Cancel (IOC); 6 = Good Till Date (GTD).</param> /// <param name="expireTime"> If supplied in the NewOrderSingle, it is echoed back in this ExecutionReport.</param> /// <param name="text">Where possible, message to explain execution report.</param> /// <param name="orderRejectionReason">0 = OrdRejReason.BROKER_EXCHANGE_OPTION</param> /// <param name="positionID">Position ID.</param> /// <returns></returns> public string ExecutionReport(SessionQualifier qualifier, int messageSequenceNumber, string cTraderOrderID, string orderStatus, string transactTime, long symbol = 0, int side = 1, int averagePrice = 0, int orderQuantity = 0, int leavesQuantity = 0, int cumQuantity = 0, string orderID = "", string orderType = "", int price = 0, int stopPrice = 0, string timeInForce = "", string expireTime = "", string text = "", int orderRejectionReason = -1, string positionID = "") { var body = new StringBuilder(); // cTrader order id. body.Append("37=" + cTraderOrderID + "|"); // Unique identifier for the order, allocated by the client. body.Append("11=" + orderID + "|"); // Execution Type. body.Append("150=F|"); // 0 = New; 1 = Partially filled; 2 = Filled; 8 = Rejected; 4 = Cancelled(When an order is partially filled, "Cancelled" is // returned signifying Tag 151: LeavesQty is cancelled and will not be subsequently filled); C = Expired. body.Append("39=" + orderStatus + "|"); // Instrument identificators are provided by Spotware. body.Append("55=" + symbol + "|"); // 1 = Buy; 2 = Sell. body.Append("54=" + side + "|"); // Time the transaction represented by this ExecutionReport occurred message(in UTC). body.Append("60=" + transactTime + "|"); if (averagePrice > 0) { // The price at which the deal was filled.For an IOC or GTD order, this is the VWAP(Volume Weighted Average Price) of the filled order. body.Append("6=" + averagePrice + "|"); } if (orderQuantity > 0) { // The fixed currency amount. body.Append("38=" + orderQuantity + "|"); } if (leavesQuantity > 0) { // The amount of the order still to be filled.This is a value between 0 (fully filled) and OrderQty (partially filled). body.Append("151=" + leavesQuantity + "|"); } if (cumQuantity > 0) { // The total amount of the order which has been filled. body.Append("14=" + cumQuantity + "|"); } if (orderType != string.Empty) { // 1 = Market; 2 = Limit. body.Append("40=" + orderType + "|"); } if (price > 0) { // If supplied in the NewOrderSingle, it is echoed back in this ExecutionReport. body.Append("44=" + price + "|"); } if (stopPrice > 0) { // If supplied in the NewOrderSingle, it is echoed back in this ExecutionReport. body.Append("99=" + stopPrice + "|"); } if (timeInForce != string.Empty) { // U1 = Good Till Cancel (GTC); 3 = Immediate Or Cancel (IOC); 6 = Good Till Date (GTD). body.Append("59=" + timeInForce + "|"); } if (expireTime != string.Empty) { // If supplied in the NewOrderSingle, it is echoed back in this ExecutionReport. body.Append("126=" + expireTime + "|"); } if (text != string.Empty) { // Where possible, message to explain execution report. body.Append("58=" + text + "|"); } if (orderRejectionReason != -1) { // 0 = OrdRejReason.BROKER_EXCHANGE_OPTION body.Append("103=" + orderRejectionReason + "|"); } if (positionID != string.Empty) { // Position ID. body.Append("721=" + positionID + "|"); } var header = ConstructHeader(qualifier, ApplicationMessageCode(ApplicationMessageType.OrderStatusRequest), messageSequenceNumber, body.ToString()); var headerAndBody = header + body; var trailer = ConstructTrailer(headerAndBody); var headerAndMessageAndTrailer = header + body + trailer; return headerAndMessageAndTrailer.Replace("|", "\u0001"); } /// <summary> /// Businesses the message reject. /// </summary> /// <param name="qualifier">The session qualifier.</param> /// <param name="messageSequenceNumber">The message sequence number.</param> /// <param name="referenceSequenceNum">MsgSeqNum<34> of rejected message.</param> /// <param name="referenceMessageType">TThe MsgType<35> of the FIX message being referenced.</param> /// <param name="businessRejectRefID">The value of the business-level 'ID' field on the message being referenced.Required unless the corresponding ID field was not specified.</param> /// <param name="businessRejectReason">Where possible, message to explain reason for rejection.</param> /// <param name="text">Where possible, message to explain reason for rejection.</param> /// <returns></returns> public string BusinessMessageReject(SessionQualifier qualifier, int messageSequenceNumber, int referenceSequenceNum = 0, string referenceMessageType = "", string businessRejectRefID = "", int businessRejectReason = -1, string text = "") { var body = new StringBuilder(); if (referenceSequenceNum != 0) { // MsgSeqNum<34> of rejected message. body.Append("45=" + referenceSequenceNum + "|"); } if (referenceMessageType != string.Empty) { // The MsgType<35> of the FIX message being referenced. body.Append("372=" + referenceMessageType + "|"); } if (businessRejectRefID != string.Empty) { // The value of the business-level 'ID' field on the message being referenced.Required unless the corresponding ID field was not specified. body.Append("379=" + businessRejectRefID + "|"); } if (businessRejectReason != -1) { // Code to identify reason for a Business Message Reject<j> message. 0 = OTHER. body.Append("380=" + businessRejectReason + "|"); } if (text != string.Empty) { // Where possible, message to explain reason for rejection. body.Append("58=" + text + "|"); } var header = ConstructHeader(qualifier, ApplicationMessageCode(ApplicationMessageType.BusinessMessageReject), messageSequenceNumber, body.ToString()); var headerAndBody = header + body; var trailer = ConstructTrailer(headerAndBody); var headerAndMessageAndTrailer = header + body + trailer; return headerAndMessageAndTrailer.Replace("|", "\u0001"); } /// <summary> /// Requests for positions. /// </summary> /// <param name="qualifier">The session qualifier.</param> /// <param name="messageSequenceNumber">The message sequence number.</param> /// <param name="requestID">Unique request ID (set by client).</param> /// <param name="positionRequestID">Position ID to request. If not set, all open positions will be returned.</param> /// <returns></returns> public string RequestForPositions(SessionQualifier qualifier, int messageSequenceNumber, string requestID, string positionRequestID = "") { var body = new StringBuilder(); // Unique request ID (set by client). body.Append("710=" + requestID + "|"); if (positionRequestID != string.Empty) { // Position ID to request. If not set, all open positions will be returned. body.Append("721=" + positionRequestID + "|"); } var header = ConstructHeader(qualifier, ApplicationMessageCode(ApplicationMessageType.RequestForPosition), messageSequenceNumber, body.ToString()); var headerAndBody = header + body; var trailer = ConstructTrailer(headerAndBody); var headerAndMessageAndTrailer = header + body + trailer; return headerAndMessageAndTrailer.Replace("|", "\u0001"); } /// <summary> /// Gets the position of reporting. /// </summary> /// <param name="qualifier">The session qualifier.</param> /// <param name="messageSequenceNumber">The message sequence number.</param> /// <param name="requestID">Id of RequestForPositions.</param> /// <param name="totalNumberOfPositionReports">Total count of PositionReport’s in sequence when PosReqResult(728) is VALID_REQUEST, otherwise = 0.</param> /// <param name="positionRequestResult">0 = Valid Request; 2 = No open positions found that match criteria.</param> /// <param name="positionID">Position ID (is not set if PosReqResult(728) is not VALID_REQUEST).</param> /// <param name="symbol">The symbol for which the current Position Report is prepared. (is not set if PosReqResult(728) is not VALID_REQUEST).</param> /// <param name="noOfPositions">1 when PosReqResult(728) is VALID_REQUEST, otherwise not set.</param> /// <param name="longQuantity">Position’s open volume in case of BUY trade side, = 0 in case of SELL trade side, is not set if PosReqResult(728) is not VALID_REQUEST.</param> /// <param name="shortQuantity">Position’s open volume in case of SELL trade side, = 0 in case of BUY trade side, is not set if PosReqResult(728) is not VALID_REQUEST.</param> /// <param name="settlementPrice">Average price of the opened volume in the current PositionReport.</param> /// <returns></returns> public string PositionReport(SessionQualifier qualifier, int messageSequenceNumber, string requestID, string totalNumberOfPositionReports, string positionRequestResult, string positionID = "", string symbol = "", string noOfPositions = "", string longQuantity = "", string shortQuantity = "", string settlementPrice = "") { var body = new StringBuilder(); // Id of RequestForPositions. body.Append("710=" + requestID + "|"); if (positionID != string.Empty) { // Position ID (is not set if PosReqResult(728) is not VALID_REQUEST). body.Append("721=" + positionID + "|"); } // Total count of PositionReport’s in sequence when PosReqResult(728) is VALID_REQUEST, otherwise = 0. body.Append("727=" + totalNumberOfPositionReports + "|"); // 0 = Valid Request; 2 = No open positions found that match criteria. body.Append("728=" + positionRequestResult + "|"); if (symbol != string.Empty) { // The symbol for which the current Position Report is prepared. (is not set if PosReqResult(728) is not VALID_REQUEST). body.Append("55=" + symbol + "|"); } if (noOfPositions != string.Empty) { // 1 when PosReqResult(728) is VALID_REQUEST, otherwise not set. body.Append("702=" + noOfPositions + "|"); } if (longQuantity != string.Empty) { // Position’s open volume in case of BUY trade side, = 0 in case of SELL trade side, is not set if PosReqResult(728) is not VALID_REQUEST. body.Append("704=" + requestID + "|"); } if (shortQuantity != string.Empty) { //Position’s open volume in case of SELL trade side, = 0 in case of BUY trade side, is not set if PosReqResult(728) is not VALID_REQUEST. body.Append("705=" + shortQuantity + "|"); } if (settlementPrice != string.Empty) { // Average price of the opened volume in the current PositionReport. body.Append("730=" + settlementPrice + "|"); } var header = ConstructHeader(qualifier, ApplicationMessageCode(ApplicationMessageType.PositionReport), messageSequenceNumber, string.Empty); var headerAndBody = header + body; var trailer = ConstructTrailer(headerAndBody); var headerAndMessageAndTrailer = header + body + trailer; return headerAndMessageAndTrailer.Replace("|", "\u0001"); } /// <summary> /// Constructs the message header. /// </summary> /// <param name="qualifier">The session qualifier.</param> /// <param name="type">The message type.</param> /// <param name="messageSequenceNumber">The message sequence number.</param> /// <param name="bodyMessage">The body message.</param> /// <returns></returns> private string ConstructHeader(SessionQualifier qualifier, string type, int messageSequenceNumber, string bodyMessage) { var header = new StringBuilder(); // Protocol version. FIX.4.4 (Always unencrypted, must be first field // in message. header.Append("8=FIX.4.4|"); var message = new StringBuilder(); // Message type. Always unencrypted, must be third field in message. message.Append("35=" + type + "|"); // ID of the trading party in following format: <BrokerUID>.<Trader Login> // where BrokerUID is provided by cTrader and Trader Login is numeric // identifier of the trader account. message.Append("49=" + _senderCompID + "|"); // Message target. Valid value is "CSERVER" message.Append("56=" + _targetCompID + "|"); // Additional session qualifier. Possible values are: "QUOTE", "TRADE". message.Append("57=" + qualifier.ToString() + "|"); // Assigned value used to identify specific message originator. message.Append("50=" + _senderSubID + "|"); // Message Sequence Number message.Append("34=" + messageSequenceNumber + "|"); // Time of message transmission (always expressed in UTC(Universal Time // Coordinated, also known as 'GMT'). message.Append("52=" + DateTime.UtcNow.ToString("yyyyMMdd-HH:mm:ss") + "|"); var length = message.Length + bodyMessage.Length; // Message body length. Always unencrypted, must be second field in message. header.Append("9=" + length + "|"); header.Append(message); return header.ToString(); } /// <summary> /// Constructs the message trailer. /// </summary> /// <param name="message">The message trailer.</param> /// <returns></returns> private string ConstructTrailer(string message) { //Three byte, simple checksum. Always last field in message; i.e. serves, //with the trailing<SOH>, //as the end - of - message delimiter. Always defined as three characters //(and always unencrypted). var trailer = "10=" + CalculateChecksum(message.Replace("|", "\u0001").ToString()).ToString().PadLeft(3, '0') + "|"; return trailer; } /// <summary> /// Calculates the checksum. /// </summary> /// <param name="dataToCalculate">The data to calculate.</param> /// <returns></returns> private int CalculateChecksum(string dataToCalculate) { byte[] byteToCalculate = Encoding.ASCII.GetBytes(dataToCalculate); int checksum = 0; foreach (byte chData in byteToCalculate) { checksum += chData; } return checksum % 256; } /// <summary> /// Returns the session the message code. /// </summary> /// <param name="type">The session message type.</param> /// <returns></returns> private string SessionMessageCode(SessionMessageType type) { switch (type) { case SessionMessageType.Heartbeat: return "0"; //break; case SessionMessageType.Logon: return "A"; //break; case SessionMessageType.Logout: return "5"; // break; case SessionMessageType.Reject: return "3"; //break; case SessionMessageType.Resend: return "2"; //break; case SessionMessageType.SequenceReset: return "4"; //break; case SessionMessageType.TestRequest: return "1"; default: //break; return "0"; } } /// <summary> /// Returns the application message code. /// </summary> /// <param name="type">The application message type.</param> /// <returns></returns> private string ApplicationMessageCode(ApplicationMessageType type) { switch (type) { case ApplicationMessageType.MarketDataRequest: return "V"; //break; case ApplicationMessageType.MarketDataIncrementalRefresh: return "X"; //break; case ApplicationMessageType.NewOrderSingle: return "D"; //break; case ApplicationMessageType.CancelOrderSingle: return "F"; // break; case ApplicationMessageType.OrderStatusRequest: return "H"; // break; case ApplicationMessageType.ExecutionReport: return "8"; // break; case ApplicationMessageType.BusinessMessageReject: return "j"; // break; case ApplicationMessageType.RequestForPosition: return "AN"; // break; case ApplicationMessageType.PositionReport: return "AP"; default: // break; return "0"; } } } }
@netread2004
netread2004
01 Nov 2018, 16:21
Any response on this?
This appears to be real issue in handshaking sent and received messages.
I don't know how one can interpret correctly the recieved messages from FIX server and take appropriate actions without proper relation between sent and recieved messages.
I assume these sent/received messages are not asynchronously exchanged. I beleive there is certain relation between sent and received messages.
Please can anyone comment on this?
@netread2004
netread2004
30 Nov 2018, 21:26
Panagiotis,
Thanks for your suggestions.
35=AF works great! I am able to get the responses for particualar 584=ID quickly unlike what used to happen with 35=H.
Also, I am now using 1 second timer to parse received messages. I can not go less than 1 second becasue Timer allows only interger type input parameter. Is there any technique one can use to invoke events like timer with time interval less than 1 sec?
.
@netread2004