开云体育

ctrl + shift + ? for shortcuts
© 2025 开云体育

ISSUES WITH SMOOTH FAILOVER IN TWS APPLICATION - IBKR SERVER CONECTION DROP.


 

ALERT LONG MESSAGE.

Hello Folks,

I am working on implementing a smooth failover mechanism to ensure the integrity of the code whenever there is a connection drop between the TWS application and the IBKR server.


Request you to provide valuable insights on below scenarios.


Case 1:


An order modification request was placed, increasing the order size beyond the previously placed quantity. The request was successfully sent to the IBKR server, but before we could receive an Order Status Callback(from that we can extract updated Order TotalQuantity), the connection between TWS and IBKR was lost.

Despite the connection drop, the order was successfully modified at the exchange.

Scenario During Connection Loss

  • If the internet was down during this period, it is possible that the order was fully executed at the exchange.

  • Once the connection is restored and we request OpenOrders, we will not receive any update related to this order because it is no longer active at the exchange.

  • However, when we request Executions, we will receive all partial or full executions for the order.

Problem with Execution Callback Data

  • The Executions Callback does not provide the remaining order size, unlike the Order Status Callback.

  • As a result, we cannot determine the actual remaining order size that was present at the exchange.

  • This forces us to rely on our internal maps, which in this case would not reflect the correct remaining size of the order.

  • Consequently, we may process extra executions, leading to discrepancies.

Example to Illustrate the Issue

  1. Initial Order Size → 10

  2. Modified Order Size → 15 (Modification was sent but confirmation was not received due to connection loss)

  3. Upon reconnection:

    • We do not receive an Order Status update for the modified order.

    • We receive a cumulative execution size of 15, but we have no way of verifying the actual remaining order size at the time of execution.

This results in an inconsistency in tracking the order status and execution details.

Currently, each time the TWS API receives an execution update, we calculate the remaining order size as follows:

remaining_order_size = remaining_order_size - current_execution_size;

if (remaining_order_size <= 0) {

????std::cout << "Order is fully executed";

????// Remove order from metadata maps

}

if(remaining_order_size<0) remaining_order_size = 0;

To ensure proper synchronization, we need to pass the remaining order size to the Execution listeners. However, in the above logic, as soon as remaining_order_size reaches zero or a negative value, the order is immediately removed from the metadata maps.

To prevent premature removal and ensure consistency, I have implemented a safeguard by updating the order size at the time of order submission. This adjustment helps maintain synchronization and prevents potential discrepancies.


Issues in using above mechanism :(

I have identified a potential issue with updating the order size at the time of order submission. Consider the following scenario:

  • A modification request is placed while the internet is down.

  • At this point, we update the order size in our system to 15 (UPDATED AT THE TIME OF SENDING THE ORDER).

  • Meanwhile, during the disconnection, the order is executed at the exchange for 10 units.

  • Once the connection is restored, we receive only the cumulative execution size of 10 from TWS (WE ASSUMED ORDER SIZE TO BE UPDATED TO 15).

  • However, since we do not receive all individual execution details, the internal system will not recognize that the order is fully executed.

As a result, the order remains in our internal maps, and we continue requesting execution details for the remaining 5 units that no longer exist.


To address this issue, we can retrieve CompletedOrders from TWS.


Any other method to handle above scenarios.?

Thanks for you patience :)


 

开云体育

Rather than comment on the questions you’ve raised, what I’ll do here is just lay out the basic principles that underly the order management component of my trading platform, which I have honed over the past 22 years. I believe it is immune to all the situations you describe.

?

  1. Maintain a model in the client containing information about all current orders. This is the client’s view of the current state of affairs, which may not be the same as the reality at any moment because of the unavoidable asynchronicity of multiple cooperating (very) loosely--coupled components.

?

  1. Record the state of the model in a persistent data store that can be used to reconstitute the model after a restart of the client (for any reason). As a matter of implementation, I use a text file containing JSON-encoded records that reflect every order-related operation or event sent to or received from the API. This ‘transaction log’ can be replayed to reconstruct the model as it was at the time the last record was written.

?

  1. Update the model whenever an order is placed, modified, or cancelled.

?

  1. Update the model whenever an open order, order status, execution or error notification is received from the API. Note that it is execution notifications that are really in control of order state; open order and order status notifications merely provide confirmation, and an inconsistency would be a definite fault (see 6 below).

?

  1. As a practical matter, the representation of orders in the model is the actual order objects used in the place order call, updated on receipt with data from open order notifications.

?

  1. If at any point an irreconcilable inconsistency emerges (for example receipt of an open order notification for an order that is no longer in the model), this is a situation that needs to be investigated (typically a programming error): execution of the client must be immediately stopped since it is not safe to proceed from an indeterminate state. (This might surprise some, but where real money is at stake it is unwise to assume that the situation will somehow correct itself.)

?

  1. Provide an optional capability to ensure that every order always has a protective stop-loss. For a manual trading client, this may not be needed, because there is a least a strong possibility that there will be a human present to assess unexpected conditions and take corrective action if need be. For an unattended auto-trading system, the situation may be very different.

?

I think that more or less covers it, though there is a lot of detail I’ve glossed over.

?

Sounds like a lot of work? It is!...

?

?

Richard

?


 

Richard nicely described how you'd use the various order and trade related callbacks to manage your orders, and there is really nothing for me to add there.

Orders and trades do not happen in the vacuum and they are always related to an Account and a Position. TWS API has a rich set of requests, subscriptions, and callbacks that keep your client informed about those as well. Every trade changes the position so that you have a "second opinion" of the exact impact of an order (or a series of orders or modifications). You can simply compare the cumulative effect of a series of execDetails() callbacks with the change of the position over the same time frame.

Below a very simplified data model of how they fit together.

闯ü谤驳别苍

PS. Generally one Trade will receive one TradeReport. But for completeness the model indicates there may be more than one. In rare situations (and I never had that happen, but it may), TWS API clients may receive one or more TradeReport corrections through execDetails callbacks later in time. The documentation says:

Note if a correction to an execution is published it will be received as an additional IBApi.EWrapper.execDetails callback with all parameters identical except for the execID in the Execution object. The execID will differ only in the digits after the final period.

?

?
?
?
?
On Fri, Feb 14, 2025 at 01:01 PM, Richard L King wrote:

Rather than comment on the questions you’ve raised, what I’ll do here is just lay out the basic principles that underly the order management component of my trading platform, which I have honed over the past 22 years. I believe it is immune to all the situations you describe.

?

  1. Maintain a model in the client containing information about all current orders. This is the client’s view of the current state of affairs, which may not be the same as the reality at any moment because of the unavoidable asynchronicity of multiple cooperating (very) loosely--coupled components.

?

  1. Record the state of the model in a persistent data store that can be used to reconstitute the model after a restart of the client (for any reason). As a matter of implementation, I use a text file containing JSON-encoded records that reflect every order-related operation or event sent to or received from the API. This ‘transaction log’ can be replayed to reconstruct the model as it was at the time the last record was written.

?

  1. Update the model whenever an order is placed, modified, or cancelled.

?

  1. Update the model whenever an open order, order status, execution or error notification is received from the API. Note that it is execution notifications that are really in control of order state; open order and order status notifications merely provide confirmation, and an inconsistency would be a definite fault (see 6 below).

?

  1. As a practical matter, the representation of orders in the model is the actual order objects used in the place order call, updated on receipt with data from open order notifications.

?

  1. If at any point an irreconcilable inconsistency emerges (for example receipt of an open order notification for an order that is no longer in the model), this is a situation that needs to be investigated (typically a programming error): execution of the client must be immediately stopped since it is not safe to proceed from an indeterminate state. (This might surprise some, but where real money is at stake it is unwise to assume that the situation will somehow correct itself.)

?

  1. Provide an optional capability to ensure that every order always has a protective stop-loss. For a manual trading client, this may not be needed, because there is a least a strong possibility that there will be a human present to assess unexpected conditions and take corrective action if need be. For an unattended auto-trading system, the situation may be very different.

?

I think that more or less covers it, though there is a lot of detail I’ve glossed over.

?

Sounds like a lot of work? It is!...

?

?

Richard

?


 

It looks like there's no way of deciding whether the order was actually modified without placing a req[All]OpenOrders call. I don't usually modify working orders but if I had to approach this problem I would likely use a circular awaitable future structure that would wait for two events to complete together: an OpenOrderEnd callback and an ExecutionEnd callback, and make a new req[All]OpenOrders call every time these two events have happened. In C# structurally this can be done either with regular events or two TaskCompletionSource objects and an await Task.WhenAll (or similar) call in a cycle. Doing this would ensure optimally timed calls to req[All]OpenOrders that would neither overlap each other nor fire needlessly in the absense of new Executions. Overall while this would inevitably lead to quite many req[All]OpenOrders calls, these will actually not require much processing power or connection hopping since these data are already in TWS so getting and processing it through the socket is going to be cheap even with some redundancy, which seems to be inevitable here
--
Best,
DS


 

In Software Engineering (and also in Automation) there is a concept called a Controller pattern. The best analogy is the clutch when you drive stick.
A controller has a:
- Resource - the object it controls
- Desired State: what it should be
- Actual State: what it currently is
The controller simply tries to issue actions to bring the actual state closer to the desired state.
?
In this case you need a double clutch as you will be driving a race car :)
?
- Position Controller
Resource: position (indexed by security)
Desired State: total # of contracts in a certain stock/security that you want to have
Actual State: sum(# of executions) + last valid position update (usually from startup/reconnect)
Action: your trading model changes the desired state only - it does not issue any orders; the position controller checks how much total qty is active in the Order Controller and taking the difference (desired vs actual - call it position error) it will issue new requests (to the Order Controller) to bring the number of active quantity in the market equal to the desired state. If it mistakenly overfills by "k" contracts (e.g. due to a bad algo in Order Controller), it will request a -k to the Order Controller. So it's self-healing to a certain degree.
?
- Order Controller
Resource: orders (indexed by security and an internal order request UUID from position controller)
Desired State: # total == # filled; you can go on a splurge with additional "desires": desired time (T), desired risk (max 4 ticks), active/passive, iceberg, VWAP etc..
Actual State: each order has # total quantity, # filled = 0 and # unfilled = # total as received from the initial request (coming from Position Controller) and then updated by subsequent API responses by adding up executions
Action: the Position Controller will ask for more exposure in the market, or will ask to reduce its exposure in the market by issuing specific requests to the Order Controller. In turn, the order controller will then issue new order(s) based on that. It could issue none initially if a desired start time is not there, etc.. Or it could issue less than the requested quantity in case of VWAP etc.. In the simple case it will just issue a single new order with total=request, filled=0, unfilled=request. A slightly more complex scenario it will modify an existing order.
As executions fill up, you update back the filled/unfilled parameters for that order id.
?
Each controller runs in it's own thread (or async routine) in parallel.
?
Now, addressing your use-case:
You request 10 desired state and you have 0 actual state in Position Controller.
Order Controller issues a new order with total qty=10 (still unfilled) associated with an internal id=0xdeadbeef.
You modify your request to 15 desired state and immediately disconnect.
Order Controller won't issue new orders or modify existing orders while disconnected.
You will have 15 in your desired state and a 0 in your actual state in the Position Controller.
You reconnect.
Once you're back up (regardless what happened in the market), you will synchronize positions and orders during startup / reconnect.
Now your Position Controller will suddenly be 15 desired state and 10 actual state.
Your Order Controller gets the list of active orders (some it knows based on internal UUID correspondence, some are maybe new, some are maybe missing).
There is 0 unfilled so initially Order Controller won't do anything. It will just remove the reference to the filled order under it's UUID 0xdeadbeef.
Next, Position Controller will issue a new request for 15-10=5 to Order Controller with an in internal id=0xaaaabbbb.
Order Controller will issue an order in the market with total qty 5.


 

I sincerely thank everyone for their valuable guidance on the design patterns I should follow to prevent discrepancies.


 

I went into the API Global Settings in TWS and IBGW today (I rarely need to go there) and that reminded me of an option you might be interested in. It allows you to control whether TWS/IBGW should remember and resubmit orders they receive while connectivity with IBKR is lost or not.
?
?
On Sun, Feb 16, 2025 at 11:17 PM, <joshi.tarun3107@...> wrote:

I sincerely thank everyone for their valuable guidance on the design patterns I should follow to prevent discrepancies.


 

I don't have this option in TWS 10.30.1t and IBGW 10.30.xx.
?
--
Best,
DS


 

Must be brand-new then. It is in TWS and IBGW 10.30.1u on Linux.
?
?
On Fri, Mar 7, 2025 at 08:39 AM, ds-avatar wrote:

I don't have this option in TWS 10.30.1t and IBGW 10.30.xx.
?
--
Best,
DS