A common reason to use the Trader Workstation API is to place an order to an Interactive Brokers’ account from a third party or custom software. This might be part of an automated strategy or caused by a manual user interaction in an API client’s graphical user interface.
In this lesson we will discuss the placeOrder function in the API EClient class, describe how it is used for order placement, as well as describing those functions which were used to monitor order stats and execution information.
Essentially, any order type which we place from TWS can also be placed from the API. This includes advanced order types such as IB Algos, bracket orders and conditional orders. Most order attributes from TWS can also be used with the TWS API. In general, order attributes must be set from the API by defining different fields within the API order class when an order is sent. However, there are also some attributes which will be read from the presets in the session of TWS or the IB Gateway to which the API client is connected.
Interactive Brokers offers an array of orders and attributes and provides access to more than 100 exchanges worldwide. This leads to numerable combinations of order types, instruments and exchanges. The best tool to use and check if a particular combination of order type, instrument and attribute is valid is in TWS itself.
Before creating a particularly complex order from the API, it is always recommended to first check if the same order can be created in TWS Then, if the combination isn’t valid, this will be indicated by TWS because either the order type will be greyed out or not displayed for the combination.
For a simple demonstration of invoking the placeOrder function and receiving the associated callbacks for monitoring orders, we will show how to place an order in the paper account for AAPL stock using a simple program.
Before placing any order to a live account, it is recommended to first place the order in the paper account to verify your order is being placed as intended.
Paper accounts are offered to all Interactive Brokers account holders and demo accounts can be accessed even before a live account is opened.
Here, I’ve created a file called placeOrder.py which demonstrates how to place the AAPL order through the API.
In this file, we will create the test class which drives from both EClient and EWrapper. But instead of overriding the functions as we did previously, we will override functions related in EWrapper for placing and monitoring orders. Namely nextValidId, orderStatus, openOrder and execDetails or execution details.
The callback nextValidID located at the top is used to receive the next valid order ID for the API client to place a new order.
The next valid ID callback is invoked in response to the API client either by calling the function’s reqIds in the EClient class or invoked after the initial connection is completed.
For this lesson, we will continue using AAPL as our contract. And while I could type out my contract object and put it directly into my order request later, this example can show off some benefit of using a combined class for EClient and EWrapper. I can use my returned contractDetails variable from EWrapper.contractDetails, and specify “contractdetails.contract” to use the contract object directly.
We will start the lesson by placing a simple market order. To get started, I will need to start a process similar to our contracts by creating an order object. I will call this “myorder”. Next, we will need to cover a few attributes that will always need to be included in our orders. I will start with myorder.orderId and set this equal to the orderId variable from the nextValidId method.
Next, I will dictate my action, using myorder.action. In this case, I will set the value to BUY, though you may enter SELL here if that is preferred. Then, I will include my order type, and set this equal to “MKT” for market order.
And then we will end by specifying my quantity. To do so, we will enter myorder.totalQuantity and set this equal to 10. And those are all the required values to place a simple order. We will discuss additional flags later, but this is simply the minimum.
Now that we have our contract object, and our order object, we are ready to move on to our order placement. Here we can make a request to EClient’s placeOrder method. To do so, I will type self.placeOrder. Then in parentheses, I will reuse my orderId as my requestId, then I will specify my contract object, then my order object.
That is all that is needed to place an order. Unlike requesting market data or contract details, we technically do not require that any EWrapper method be called. I can look in Trader Workstation and see that this order has been placed.
However, I think we may want to see details on our orders returned to us after we submit them. Let’s first create a limit order to better showcase these fields.
I will keep most of my order object the same as before, with a few key exceptions. First, I will change my action to “SELL” instead of “BUY”.
After that I will move on to Time in Force, or TIF. I will be using “GTC”, though this field has the same options as TWS, such as MOC for market on close. I would note that if no TIF is specified, it will use DAY order by default.
Then, I can change my order type to LMT instead of MKT.
And finally, I will use myorder.lmtPrice to specify my limit value. For AAPL, I will specify 144.80
As we could see from our contract details request, all contracts have a minimum tick value. For AAPL stock, for instance, this is one cent min tick, so orders can’t have prices specified with more than two decimal places or less than one cent. There is more information about finding the minimum increment for different instruments in the API reference guide.
Now that we have the order ID, the contract object and the order object, we can invoke place order. After TWS receives a valid order, it will begin sending back messages to the callback functions, order status, open order and execution details to indicate an order status as well as any changes in the order status, which occur.
Open Orders
The openOrder method will show us details whenever an order is placed. We will see the full order and contract details of a given order after it has been placed. For instance, in this case we would expect to see an AAPL limit order there with a time in force of GTC and an order price of 144.80 USD.
It is important to note that openOrder also returns the orderState object. orderState will include values such as commission, initial margin, and maintenance margin. Just like we did with contractDetails before, we can review the order_state.py file in the source code to see what other flags can be requested specifically.
Now, in addition to the openOrder function, we will also want to include the orderStatus method. While openOrder provides information on an order once it is placed, orderStatus provides information on the order thereafter. After every submission of an order, as soon as the order is valid usually you would see an order status callback which would be either pre-submitted, submitted or filled. Sometimes there will be duplicates which might need to be filtered out by the API client.
Order Executions
If the order executes there will always be execDetails or execution details for every partial fill. This can be thought of as the summary of the order. We can review the orderId, the executionId, information on the contract, information on the average price, and so on.
So, if there is more than one fill for a particular order you would see multiple callbacks for exec details. The callback would have information such as the order ID, which would be different than the API order ID. This is an order ID which is unique in the account and generated by TWS.
You would see the number of shares which have executed. You would also see a last liquidity flag which would tell us if this execution either added or removed liquidity from the market.
I will go ahead and run the script and that should immediately place the order for AAPL stock. A sell order for ten shares at the limit price of 144.00.
When we scroll down to the bottom, we can see that we receive an immediate execution. That’s because the current price of AAPL is at about 144.75 and this is during market hours, so the order immediately became active.
Before that occurred, the callback we received was open order. Open order callback let us know the details of the order.
The next orderStatus we receive will be a filled status because the order filled pretty much immediately, at least in the paper trading simulation engine and that tells us how many shares filled, how many are remaining, and also the fill price.
We also received execution details more or less simultaneously with that order status so there’s information like the exact execution details ID as well as the same information about the number of shares which filled and the filled price.
Modify Orders
Now let’s say we want to modify an order. Because we included the EWrapper callbacks for our order, we can receive our order ID, and easily modify this order whenever we need. We can simply return to our order object with our problematic orderId, change our orderId from nextValidId’s orderId, and use it here. Now, any changes applied to this order submission will modify our original order.
While I may be using a very simple example here with very human error, I would like to emphasize that this process can all be done programmatically for on-the-fly adjustments to your orders to coincide with the market, or other more complex examples.
Let’s go ahead and walk through this process. Let’s say I placed an order for $1490.00 instead of my $149.00 example. Just a simple typo with very big repercussions all because I missed the decimal place. Because I have the orderId from that request in my callback from before, I can modify this order. I can change my order to use orderId 15 again, then put in the correct price. Now, if I go ahead and put $149.00 in my limit price value and then resubmit the order, I can run my code again, and we can see that this order updated. And if you had any doubts, we can return to Trader Workstation and see that this order has updated there as well.
These are just some order types that Interactive Brokers offers through the Trader Workstation, and you are welcome to explore our wide array of available offerings notated in our documentation.
This concludes our video on Basic Orders and Order Modification in the TWS API. Thank you for watching, and we look forward to having you join us for more TWS Python API lessons.
Available Order Types & Algos using TWS API
Retrieving current active Orders
from ibapi.client import * from ibapi.wrapper import * class TestApp(EClient, EWrapper): def __init__(self): EClient.__init__(self, self) def nextValidId(self, orderId: OrderId): mycontract = Contract() mycontract.symbol = "AAPL" mycontract.secType = "STK" mycontract.exchange = "SMART" mycontract.currency = "USD" self.reqContractDetails(orderId, mycontract) def contractDetails(self, reqId: int, contractDetails: ContractDetails): print(contractDetails.contract) myorder = Order() myorder.orderId = reqId myorder.action = "SELL" myorder.tif = "GTC" myorder.orderType = "LMT" myorder.lmtPrice = 144.80 myorder.totalQuantity = 10 self.placeOrder(myorder.orderId, contractDetails.contract, myorder) def openOrder(self, orderId: OrderId, contract: Contract, order: Order, orderState: OrderState): print(f"openOrder. orderId: {orderId}, contract: {contract}, order: {order}") def orderStatus(self, orderId: OrderId, status: str, filled: Decimal, remaining: Decimal, avgFillPrice: float, permId: int, parentId: int, lastFillPrice: float, clientId: int, whyHeld: str, mktCapPrice: float): print(f"orderId: {orderId}, status: {status}, filled: {filled}, remaining: {remaining}, avgFillPrice: {avgFillPrice}, permId: {permId}, parentId: {parentId}, lastFillPrice: {lastFillPrice}, clientId: {clientId}, whyHeld: {whyHeld}, mktCapPrice: {mktCapPrice}") def execDetails(self, reqId: int, contract: Contract, execution: Execution): print(f"reqId: {reqId}, contract: {contract}, execution: {execution}") app = TestApp() app.connect("127.0.0.1", 7497, 100) app.run()
Hello,
how can I place an order outside the regular trading hours.
Thank you very much.
Thank you for asking, Manfred. To submit an order to work outside of regular trading hours (Outside RTH) you can either specify it at the time of order creation or define it as an order preset to be used automatically when available. Please review this step-by-step FAQ on how to trade outside RTH on TWS: https://www.ibkr.com/faq?id=27271538. It is also possible to make these orders in Client Portal (https://www.ibkr.com/faq?id=102543637) and IBKR Mobile (https://www.ibkr.com/faq?id=29088910). We hope this helps!
If your question is specific to an API and this response did not answer it completely, please create a web ticket. The best category to choose is “API.” Our API experts will be able to help you out from there!
How can i place order for multiple clients ?
Hello Prasen, thank you for reaching out. You can place orders for multiple clients in TWS using pre-defined Account Groups. For steps on how to do this, please visit this page in our user guide: https://www.ibkrguides.com/tws/usersguidebook/financialadvisors/create%20an%20order%20for%20multiple%20clients.htm? If your question is specific to an API and this response did not answer it completely, please create a web ticket. The best category to choose is “API.” Our API experts will be able to help you out from there!
Hello, I put this example code to teste and I received this following error:
‘TypeError: EClient.__init__() missing 1 required positional argument: ‘wrapper’.
Can you help please?
Hello Paulo, we appreciate your question. For this inquiry, please open a web ticket in Client Portal (click the “Help?” in the upper right menu, then “Secure Message Center”). The best category to choose is “API.” Our API experts will be able to help you out from there!
hi! while using def displayGroupUpdated(self, reqId: int, contractInfo: str):
contract info will give me a number@exchange 8314 for IBM how do i translate the 8314 to IBM ?
or is there a way to just create an order with the unique
number for the corresponding stock?
thanks and have a good one !
Hello Snir, we appreciate your question. For this inquiry, please open a web ticket in Client Portal (click the “Help?” in the upper right menu, then “Secure Message Center”). The best category to choose is “API.” Our API experts will be able to help you out from there!
Hi,
I get this error:
Name “Decimal” is not found! How can I resolve this?
Is there a way to place orders keep track of our balance over time on the paper account?
Daniel, for this inquiry, please open a web ticket in Client Portal (click the “Help?” in the upper right menu, then “Secure Message Center”). The best category to choose is “API.” Our API experts will be able to help you out from there!
Hi Daniel, For the error: “Name “Decimal” is not found!”, please check whether your computer has the “_decimal.pyi” file. In my computer and I am using Visual Studio Code, the file path is this: C:\Users\cpoon\.vscode\extensions\ms-python.vscode-pylance-2024.6.1\dist\typeshed-fallback\stdlib\_decimal Please see whether there is “_decimal.pyi” file in your computer by searching the above similar File path. The “Decimal” comes from the `_decimal.pyi` file built by Python ORG. Here is the “_decimal.pyi` file code open sourced by Python ORG: https://github.com/python/typeshed/blob/main/stdlib/_decimal.pyi When you install Python, the “_decimal.pyi” file should be automatically installed. And then when you use Python and call “Decimal”, the Python interpreter should automatically read the “Decimal” from your local “_decimal.pyi” file.
Hello,
maybe not the right topic, but after a successful verification with IB (of epic proportions :)), I would like to create some kind of RSI based applications with DCA trading method, with which we determine the deal start condition according to the RSI. I am actually interested in whether such applications or other trading strategies can be implemented.
Thanks for the reply.
Best regards
Good question, Miroslav. Please direct this question to our API experts by creating a web ticket in Client Portal with “API” as the category.
Hello, I am trying to run the code as posted above, but am receiving the error:
File “c:\TWS API\source\pythonclient\ibOrder.py”, line 30, in
app = TestApp()
^^^^^^^^^
TypeError: EClient.__init__() missing 1 required positional argument: ‘wrapper’
Could you advise at all what I might change? Thank you!
Hello CMH, we appreciate your question. For this inquiry, please open a web ticket in Client Portal (click the “Help?” in the upper right menu, then “Secure Message Center”). The best category to choose is “API.” Our API experts will be able to help you out from there!
Same problem:
“Hello, I put this example code to teste and I received this following error:
‘TypeError: EClient.__init__() missing 1 required positional argument: ‘wrapper’.
Can you help please?”
–> General hint for this?
Hello Thomas, we appreciate your question. For this inquiry, please open a web ticket in Client Portal (click the “Help?” in the upper right menu, then “Secure Message Center”). The best category to choose is “API.” Our API experts will be able to help you out from there!
How to reset OrderID in my simulation account?
Hello David, thank you for asking. I am not sure what you mean by “OrderID,” however, you cannot reset your username or account number. The only thing you can reset is the dollar amount in the simulation account. Please review this FAQ for more information: https://www.ibkr.com/faq?id=32644862
Hi, I tried placing order thru IB gateway but it showed error: can’t write – with server and client version 156. I am using linux and the latest IB gateway api build 10.25, settings and pre
However, when I tried it on windows – same setup. It works and it showed server and client version 172.
May I know why is there a difference in the server and client versions between the two different OS? and if this is the reason why I wasn’t able to place order?
Hello Calvin, we appreciate your question. There is a chance you have not updated your API directly, or you are operating on Python and have not run the script to update your environment. Our API experts would be glad to help you out! Please create a web ticket for this inquiry and select “API” as the category: https://www.interactivebrokers.com/sso/resolver?action=NEW_TICKET. We hope this helps!
Hello,
These two topics are not discussed in the video, is there a separate video?
Checking Margin Changes
Order Efficiency Ratio (OER)
Thanks,
Sam
Hello Sam, thank you for reaching out. Resources on Checking Margin Changes and Order Efficiency Ratio (OER) are listed below.
Checking Margin Changes:
https://interactivebrokers.github.io/tws-api/margin.html
Order Efficiency Ratio:
https://interactivebrokers.github.io/tws-api/order_limitations.html
We hope this helps!
How can I place order through TWS API of an FA Account for a single client?
Hello Jiabang, please read this page in the Users’ Guide for instructions on placing an order for a Single Managed Account. We hope this helps! https://interactivebrokers.github.io/tws-api/financial_advisor_methods_and_orders.html
How to place order quantity beyond exchange freeze quantity ?
Max 50 orders per second is allowed , i would like to place more than 50X Freeze qty
Hello Sanjay, thank you for reaching out. Freeze quantity is a limitation of each exchange and is not something that can be resolved through the API or Interactive Brokers.
I’m attempting to place a trailing stop order either for long position or short position depending on trading parameters. I’m using the trailingpercent method in the code and wanted to know if using the trailing stop on a short position will work? I had a simulated short sell order with a trailing stop and the stock price was $213 but when I looked at the trailing stop which should have been a “BUY” because I was short selling, it appeared the trailing “BUY” was lower then stock purchase price; which would be incorrect because it should be the inverse on a short sell. Can you provide guidance on this?
Hello, thank you for reaching out. For users looking to trade with Trailing Stop Limit orders, you will need to include both the “price” and “auxPrice” fields to complete a successful order. You can find an example of this order here under the CURL tab ( https://www.ibkrcampusdev.wpengine.com/campus/ibkr-api-page/order-types/#trailing-stop-limit-order ). We are looking to expand our available tutorials and will look to add the trailing orders in our expansion. Thank you for your feedback.
How to identify the list of open orders that can be canceled? Can I cancel a voidable order, based on its name and order type?
I can’t find anything on the internet. Can you give me the code in detail?
THANKS
Hello, thank you for reaching out. Generally speaking, calling EClient.reqAllOpenOrders() or EClient.reqOrderStatus() will return the current status of an order, be it submitted, canceled, filled, or otherwise. You can reference this status if the order can be canceled or not. Orders can only be canceled based on orderId. Please reach out via web ticket with any more questions or concerns: https://www.interactivebrokers.com/sso/resolver?action=NEW_TICKET. Our API experts would love to guide you!
I followed the instructions on trailing stop limit orders. Therefore, I defined action, orderType, totalQuantity, trailStopPrice, limitPriceOffset and the trailing amount. I placed several orders in my paper trading account via API and manually in the TWS and played with the values. In the end there was no difference in outcome. Sooner or later every order was executed at the original stop price **without any trailing** and after all gains were eaten up. Either I miss something e. g. in the settings or there is a bug on IB’s side. Please advice. Thank you in advance.
Hello, thank you for reaching out. For assistance with personal execution behavior and understanding, please reach out to our Trade Issues team. They can be contacted through a Web Service Ticket ( https://www.ibkrguides.com/complianceportal/creatingaticket.htm ) or by calling in directly through your local line (https://www.interactivebrokers.com/en/support/customer-service.php?p=contact), and using menu options 1 and 2. We hope this helps!
Hi, how can I place order for overnight? Thanks.
Hello, we appreciate your question. Both of Interactive Brokers’ APIs support OVERNIGHT trading.
In the case of TWS API, you must use TWS and the API Latest releases (10.26 and above). Then, you must specify the OVERNIGHT exchange while submitting the order. One example of this contract would look like this:
contract = Contract()
contract.symbol = “AAPL”
contract.secType = “STK”
contract.currency = “USD”
contract.exchange = “OVERNIGHT”
contract.primaryExchange = “NASDAQ”
The client portal API is similar, in that the listingExchange must be specified in the same manner. Any example order body to mirror the above AAPL contract would be something like this:
{
“conid”:265598,
“listingExchange”: “OVERNIGHT”
}
We hope this helps!
HOW TO TRADE THE ORDER LIKE SPX DAY OPTION
Thank you for reaching out. In this lesson, we discussed order placement. To specify which contracts you will place orders for, such as SPX, I would encourage you to review our Essential Components of TWS API Programs lesson. We hope this helps!
when you place an order, in your example, why didn’t you add the account where the order will be allocated in to?
Hello, thank you for reaching out. In a single account structure, such as the one we used in our demonstration, the current account identifier is automatically assumed and unnecessary. An account ID will only need to be specified when using an advisor account that holds several sub-accounts, or in a multi-account structure where your username is tied to more than 1 account.
We hope this helps.
Hey is it possible to create order method with use and not number of shares for the total quantity parameter? for example order = Order() order.action = ‘BUY’ order.type = ‘MOC’ order.totalQuantity = 2000 (Quantity in USD instead of shares)
Hello Ravi, we appreciate your question. The API offerings do not support Cash Quantity orders for trading any security type aside from Crypto at this time. Orders must be submitted using shares instead. If you have any additional questions, please create a web ticket for this inquiry; we have a category specifically for “API.” One of our API experts will be happy to guide you! http://spr.ly/IBKR_ClientServicesCampus
The reality is that the election is right around the corner and many votes wont even happen. Get out and vote.
Hello,
the piece of code above runs smoothly, but I was wondering:
why do you use app.run? The script keeps running indefinitely and I cannot find a way to stop it. It was my understanding that launching a thread is required when running an application in tws, but I do not see it here.
I hope I explained myself.
Thank you
Hello, thank you for reaching out. App.run is essential for operating the EWrapper loop and managing the internal EReader thread to transmit data back and forth over the socket to Trader Workstation. While threading is certainly important, and largely mandatory for applications of scale, however, individual requests like this can be handled without implementing threading. This could be constructed the same as our prior lessons where we thread the app.run function. If you are interested in a more scaled model of trading, you may want to review Lesson 11 which goes into detail about how to run multiple functions to better show off scenarios where threading provides the most value. We hope this helps answer your question!
Hello Team,
I got below error while trying to place the order
ERROR 1 10268 The ‘EtradeOnly’ order attribute is not supported.
Thank you for reaching out. This error indicates that your system is not meeting the correct requirements. Please ensure you have downloaded the most up-to-date version of the TWS API. Please view this IBKR Campus course for instructions: https://www.interactivebrokers.com/campus/trading-lessons/accessing-the-tws-python-api-source-code/
We hope this helps!