Disclaimer: This project is not affiliated with, endorsed by, or associated with The Charles Schwab Corporation. All registered trademarks are the property of their respective owners. Use of these names, logos, and brands is for identification purposes only. This project is licensed under the MIT license, and acts in accordance with Schwab's API terms and conditions.
 Â
Two Streaming Dashboards From The examples directory
schwab-client-js gives you complete access to the Schwab REST API using convenient classes and methods. You can stream real-time market data, create and track orders, and retrieve information about your account as well as retrieve different types of market data.
Note: This project only supports nodejs. Although technically you could tweak schwab-client-js to run in a web browser, security concerns make that a poor choice.
Software prerequisites: nodejs version 18 or newer and a nodejs package manager such as yarn or npm
Install the package:
$ npm install schwab-client-js
or
$ yarn add schwab-client-js
- You need a Schwab brokerage account. The account is free. There is no minimum balance requiremnent.
- Login to your Schwab account and turn on thinkorswim. If thinkorswim is already enabled, you won't see instructions on how to enable it.
- You need to signup for a Schwab developer account. The account is free.
- Login to your developer account and create an "app" which is really just a web page where you configure the metadata for your API calls.
See the Schwab API Configuration doc for details on creating your Schwab app.
- You have to wait (probably 1-3 days) for your app to be approved. Go to your dashboard to check on the status of your app. You cannot use the API while the status of your app is Approved - Pending. When the status of your app is Ready For Use, you can proceed.
- Once your app is approved, click on View Details for your app. At the bottom you should see your App Key and Secret.
Schwab uses three-legged OAuth for authentication. The details are on developer.schwab.com here and here.
schwab-client-js uses environment-based security e.g. it uses a .env
file to store your security tokens.
[Note: For situtations where using a .env
file or environment variables may not be optimal (possibly AWS Lambda, for example), schwab-client-js also supports injecting your security tokens directly e.g. const mktclient = new MarketApiClient(appKey,appSecret,appRefresh);
]
- Create a
.env
file at the root of your project and add the App Key and Secret from your app on developer.schwab.com. Optionally, you can add your callback URL to your .env file like this (for details on creating yourSCHWAB_CALLBACK_URL
, see the Schwab API Configuration doc):
SCHWAB_CALLBACK_URL=https://127.0.0.1:5556
SCHWAB_APP_KEY=xxxxxxxxxxxxxxxxxxxxxxxxxxxxx
SCHWAB_SECRET=zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz
If you don't specify a SCHWAB_CALLBACK_URL
, I will assume a default value of https://127.0.0.1:5556
- Creating a
SCHWAB_REFRESH_TOKEN
. Runschwab-authorize
ORmanual-authorize
to help you create theSCHWAB_REFRESH_TOKEN
. You should be able to runschwab-authorize
from the root of your project (on MacOS and Linux) by merely typingschwab-authorize
. On Windows, you'll likely have to run it with node and specify the full path:C:\> node node_modules/schwab-client-js/bin/schwab-authorize.js
Same withmanual-authorize
. The scriptmanual-authorize
is for users who would rather not deal with the self-signed SSL certificate thatschwab-authorize
generates.
See the Schwab API Configuration doc for details about schwab-authorize
and manual-authorize
.
- Your
.env
file should now look like this (as previously mentioned, theSCHWAB_CALLBACK_URL
is optional and will default tohttps://127.0.0.1:5556
if not provided):
SCHWAB_CALLBACK_URL=https://127.0.0.1:5556
SCHWAB_APP_KEY=xxxxxxxxxxxxxxxxxxxxxxxxxxxxx
SCHWAB_SECRET=zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz
SCHWAB_REFRESH_TOKEN=yyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyy
The SCHWAB_REFRESH_TOKEN
lasts for seven days. It then expires and you will no longer be able to make API calls. You must run schwab-authorize
OR manual-authorize
again to generate a new SCHWAB_REFRESH_TOKEN
(sorry folks, I don't make these rules).
schwab-client-js uses classes and methods to make it easy to develop apps using the Schwab API. For details, see the schwab-client-js developer reference guide
schwab-client-js has three classes that you use to make API calls, and some helper functions that you can use to help create input parameters. The three classes are:
MarketApiClient()
- Get market dataTradingApiClient()
- Execute trades and get your personal account and trade-related dataStreamingApiClient()
- Stream market data
There are also about 30 helper functions that help you create JSON trade objects and option symbols. See the schwab-client-js developer reference guide for more details
Getting Market Data (see the examples directory for more examples)
import { MarketApiClient } from "schwab-client-js";
const mktclient = new MarketApiClient();
// Get stock information by ticker symbol
const data1 = await mktclient.quoteById("AMD", "quote,fundamental");
console.log("quoteById DATA=", JSON.stringify(data1));
// Get security information by cusip
const data2 = await mktclient.instrumentsCusip("023135106");
console.log("instrumentsCusip DATA=", JSON.stringify(data2));
// Get market hours information
const data3 await mktclient.markets("bond,option,forex");
console.log("markets DATA=", JSON.stringify(data3));
Helper Functions For Creating Order Objects And Option Symbols (see the examples directory for more examples)
schwab-client-js has various helper functions to make is easier to create order objects and option symbols.
// Create a Schwab-formatted option symbol
import { optionSymbol } from "schwab-client-js/orderhelp";
const optsymbol = optionSymbol("XYZ", "210115", "C", "62.50");
// optsymbol is now: "XYZ 210115C00062500"
// Returns a pre-filled JSON order object for an equity buy-to-cover market order.
import { equityBuyToCoverMarket } from "schwab-client-js/orderhelp";
const orderObj = equityBuyToCoverMarket("TSLA", 38);
console.log(JSON.stringify(orderObj, null, 2));
The output is:
{
"orderType": "MARKET",
"session": "NORMAL",
"duration": "DAY",
"orderStrategyType": "SINGLE",
"orderLegCollection": [
{
"instruction": "BUY_TO_COVER",
"quantity": 38,
"instrument": {
"symbol": "TSLA",
"assetType": "EQUITY"
}
}
]
}
orderObj
can be used as input to the placeOrderByAcct()
method as shown in the section below.
Doing Trades And Fetching Trading Data (see the examples directory for more examples)
import { TradingApiClient } from "schwab-client-js";
const trdclient = new TradingApiClient();
// Get your authorized Schwab account numbers and hashed account numbers
const data1 = await trdclient.accountsNumbers();
console.log("accountsNumbers DATA=", JSON.stringify(data1));
// Get orders (all statuses) withing the specified date range
const accountHash="4B9A9B50B7886A574E2A793DFE9B944EA2DAB9"; // Your hashed account number
const fromDate="2024-01-10T12:17:41+02:00";
const toDate="2024-09-10T12:17:41.000Z";
const data2 = await trdclient.ordersByAccount(accountHash, fromDate, toDate);
console.log("ordersByAccount DATA=", JSON.stringify(data2));
// Create a limit order during normal trading hours to buy
// one share of CTRN (Citi Trends Inc) stock for $1 USD
const orderObj = {
"orderType": "LIMIT",
"session": "NORMAL",
"duration": "DAY",
"orderStrategyType": "SINGLE",
"price": "1.00",
"orderLegCollection": [
{
"instruction": "BUY",
"quantity": 1,
"instrument": {
"symbol": "CTRN",
"assetType": "EQUITY"
}
}
]
};
// Place the above trade using the specified Schwab account
const accountHash="4B9A9B50B7886A574E2A793DFE9B944EA2DAB9"; // Your hashed account number
const data3 = await trdclient.placeOrderByAcct(accountHash, orderObj);
console.log("placeOrderByAcct DATA=", JSON.stringify(data3)); // Should return an orderId
// Delete an existing order
const accountHash="4B9A9B50B7886A574E2A793DFE9B944EA2DAB9";
const orderID="435234523452345";
const data4 = await trdclient.orderDelete(accountHash, orderId);
console.log("orderDelete DATA=", JSON.stringify(data4)); // Successful deletes return null so this should show no JSON
Streaming (see the examples directory for more examples)
- Streaming may not work well outside of normal market hours. Use the
marketById()
ormarkets()
calls to get the market hours.
import { StreamingApiClient } from "schwab-client-js";
// Create your streaming client object
const streamclient = new StreamingApiClient();
// Listen for the "open" request
streamclient.streamListen('open', () => {
console.log("Received open message: webSocket connection opened.");
});
// Listen for messages from Schwab. This is where your data arreves
streamclient.streamListen('message', (message) => {
console.log("Received data message:", message);
});
// Listen for close
streamclient.streamListen('close', (code, reason) => {
console.log(`Connection closed: Code=${code}, Reason=${reason}`);
});
// Listen for errors
streamclient.streamListen('error', (error) => {
console.error("Received error message:", error);
});
// Initialize the stream
await streamclient.streamInit();
// Login to the Schwab stream
await streamclient.streamSchwabLogin();
// Subscribe to some stock prices
let params = { keys: "AMD,TSLA,GOOG", fields: "0,1,2,3,4,5" };
await streamclient.streamSchwabRequest("ADD", "LEVELONE_EQUITIES", params);
// Streaming data happens.......
// Close the Schwab stream
await streamclient.streamSchwabLogout()
streamclient.streamClose();
The examples directory has three different streaming dashboards
- schwab-dashboard-react A React-based stock dashboard using recharts
- schwab-dashboard-forex A pure HTML/CSS forex dashboard using lightweight-charts
- schwab-dashboard-nextjs A NextJS/React/Tailwind stock dashboard using recharts
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.