Paste a target-weights CSV, preview the rebalance, execute it on your brokerage account. Tastytrade + Alpaca + Tradier + IBKR + Schwab + Hyperliquid + paper.
Project description
msts-trader
Paste a target-weights CSV, preview the rebalance, execute it on your own brokerage account. Multi-broker, local-only, no key custody.
7 brokers (Tastytrade, Alpaca, Tradier, IBKR, Schwab, Hyperliquid, paper),
leverage + margin-aware sizing (real broker margin, on by default),
sells-before-buys, optional protective stops (stop_pct column, 6/7
brokers), multi-account, headless (cron / GitHub Actions),
notifications, idempotency, and a --json API. Licensed PolyForm
Noncommercial.
$ msts-trader
Paste CSV (ticker,weight), then Ctrl+D:
ticker,weight
SPY,0.42
GLD,0.18
SHV,0.20
EEM,0.20
^D
✓ loaded 4 targets.
tastytrade · account 5W****** · NAV $48,213.42 · cash $2,150.00 · BP $46,290.00
Market: open · closes in 23 min
Rebalance preview
┃ Symbol ┃ Current % ┃ Target % ┃ Δ $ ┃ Action ┃ Note ┃
┃ SPY ┃ 18.2% ┃ 42.0% ┃ +$11k ┃ BUY 22.00 @ ~$521.34 ┃ ┃
┃ EEM ┃ 31.5% ┃ 20.0% ┃ -$5k ┃ SELL 119.00 @ ~$47.21 ┃ ┃
...
Execute 4 orders on tastytrade? [y/N]: y
[1/4] SPY BUY 22.00 @ MKT ... ROUTED id=4f8...
Done. tastytrade: sent 4, failed 0
Supported brokers
| Broker | Status | Auth | Install |
|---|---|---|---|
| Paper | shipped, tested | local file | built-in |
| Tastytrade | shipped, live-tested | OAuth refresh token | built-in |
| Alpaca | shipped, live-tested | API key + secret | built-in |
| Tradier | shipped, beta | bearer token (REST) | built-in (free sandbox to test) |
| IBKR | shipped, live-tested | TWS / IB Gateway socket | pip install "msts-trader[ibkr]" |
| Schwab | shipped, beta | OAuth2 + browser callback | pip install "msts-trader[schwab]" |
| Hyperliquid | shipped, experimental | API-wallet private key | pip install "msts-trader[hyperliquid]" |
- Live-tested = connect / balances / positions / quotes / order path verified against a real account (Tastytrade & Alpaca filled real 1-share orders; IBKR verified read + dry-run).
- Beta (Schwab, Tradier) = parsing logic is unit-tested (Tradier
against mocked HTTP) but no live fill confirmed by the author. Tradier
has a free sandbox (
TRADIER_SANDBOX=1) — easy to verify yourself. - Experimental (Hyperliquid) = crypto perps DEX; the adapter is built
on the public SDK but has not been run against a live account. Test on
testnet (
HL_TESTNET=1) with tiny size first.
IBKR + EU accounts: an EU-regulated IBKR account cannot trade US-domiciled ETFs (KID/PRIIPs, Error 201). US stocks may still be cancelled by an account Order Preset (Error 10349 → fix in TWS Global Configuration → Presets). Tastytrade and Alpaca have neither limit.
Open a GitHub issue to prioritise a broker (Tradier and a ccxt-based crypto adapter are likely next).
Install
pip install msts-trader
or with uv (installs the CLI into an isolated environment, no venv juggling):
uv tool install msts-trader
Python ≥3.11 required (uv fetches a suitable Python automatically).
Optional brokers
IBKR and Schwab require extra dependencies. Install them only if you plan to use that broker:
pip install "msts-trader[ibkr]" # adds ib_insync + nest_asyncio
pip install "msts-trader[schwab]" # adds schwab-py
pip install "msts-trader[hyperliquid]" # adds hyperliquid-python-sdk + eth-account
pip install "msts-trader[all]" # everything
(with uv: uv tool install "msts-trader[all]")
Note (IBKR + uv tool, versions ≤ 0.12.0):
uv tool installpicks the newest Python it can find (currently 3.14), where IBKR auth in older releases failed with a "no current event loop" error fromib_insync/eventkit. Fixed in releases after 0.12.0; if you're stuck on an older version, pin Python 3.13:uv tool install --python 3.13 --reinstall "msts-trader[all]"
uv runfrom a source checkout was never affected — it honors the .python-version pin.
Install from source:
git clone https://github.com/markudevelop/msts-trader.git
cd msts-trader
pip install -e ".[all]"
or with uv — uv sync creates the venv, pins Python to
.python-version, and installs everything:
git clone https://github.com/markudevelop/msts-trader.git
cd msts-trader
uv sync --all-extras
uv run msts-trader --help
One-time setup
You provide your own broker credentials. They are stored in your OS keychain (macOS Keychain / Windows Credential Manager / libsecret on Linux) and never leave your machine.
Tastytrade
- Sign in at https://developer.tastytrade.com → My Apps
- Create an OAuth application — copy the provider secret
- Run their OAuth authorization flow to obtain a refresh token
- Look up your account number in the Tastytrade web dashboard (optional)
- Run:
msts-trader login --broker tastytrade
Using Tastytrade's certification (sandbox) environment? Cert-issued
keys are rejected by production (and vice versa) — set TT_TEST=1 (env
or creds file) so msts-trader connects to the cert API instead.
Alpaca
- Sign in at https://alpaca.markets (paper or live)
- Account → API keys → generate a new pair
- Run:
msts-trader login --broker alpaca
You choose paper vs live at login time.
Tradier
msts-trader login --broker tradier
Get an access token at https://developer.tradier.com — a free sandbox
token works for end-to-end testing. Your account number is
auto-discovered if you leave it blank. Choose sandbox or production at
login. Headless: TRADIER_ACCESS_TOKEN / TRADIER_ACCOUNT_ID /
TRADIER_SANDBOX.
IBKR
pip install "msts-trader[ibkr]"
msts-trader login --broker ibkr
On versions ≤ 0.12.0 installed via uv tool, use --python 3.13 — see
the install note about IBKR on Python 3.14.
You'll be asked for host, port, and client id of a running TWS or IB Gateway. Defaults:
- TWS live:
127.0.0.1:7496 - TWS paper:
127.0.0.1:7497 - Gateway live:
127.0.0.1:4001 - Gateway paper:
127.0.0.1:4002 - Dockerised Gateway: usually
127.0.0.1:4002(whatever you mapped)
Before logging in, enable Configure → API → Enable ActiveX and Socket Clients in your TWS / Gateway. msts-trader connects, lists your managed accounts, and confirms NAV.
Schwab
pip install "msts-trader[schwab]"
msts-trader login --broker schwab
Requires a Schwab Developer app (https://developer.schwab.com) with the
callback URL set to https://127.0.0.1:8182. msts-trader pops a
browser window, you authorize, and the token JSON is written to
~/.msts-trader/schwab_token.json. Schwab refresh tokens expire every
7 days — re-run msts-trader login --broker schwab when that happens.
The callback URL must match your app's registration EXACTLY — character for character, trailing slash included. Schwab treats
https://127.0.0.1:8182andhttps://127.0.0.1:8182/as different URLs: a mismatch shows an error page on schwab.com during authorization, or fails the flow afterwards with "authorization failed or the token expired". If your app is registered with a different callback (port, slash, …), enter that exact value at the login prompt or setSCHWAB_CALLBACK_URL.
Don't wait for it to expire mid-week: run
msts-trader login --broker schwab --reauth
on a Saturday or Sunday to force a fresh browser authorization and restart the 7-day clock, guaranteeing auth works through the whole trading week.
Paper (offline simulator)
msts-trader login --broker paper
No real money, no broker connection. The book persists in
~/.msts-trader/paper_state.json between sessions. Reset any time with
msts-trader paper-reset.
The first login you complete becomes the default broker. Override per
command with --broker NAME, or change the default by logging in again.
Daily usage
-
Get your CSV. Click Copy CSV on the supported weights site, or build your own:
ticker,weight,stop_pct SPY,0.42, GLD,0.18,0.05 EEM,0.20, SHV,0.20,
weightis a fraction of NAV (e.g.0.42= 42%), not a percent.- Sum ≤ 1.0 holds the remainder as cash; sum > 1.0 is leverage
(e.g.
1.60= 160% gross, financed on margin — see Leveraged weights). - No shorts: negative weights are rejected.
stop_pctis optional — a protective-stop column. See Protective stops.- Comments starting with
#are ignored (and# asof: <iso>enables the stale-CSV guard).
-
Run:
msts-trader # uses default broker
msts-trader --broker alpaca # explicit broker
- Paste the CSV, hit
Ctrl+D(Ctrl+Zthen Enter on Windows). - Review the preview carefully.
- Type
yto execute, anything else to cancel.
Useful flags
msts-trader rebalance --dry-run # preview only, never sends
msts-trader rebalance --yes # skip the confirm prompt
msts-trader rebalance --threshold 0.02 # tighter rebalance (default 4%)
msts-trader rebalance --csv-file targets.csv # read from a file
msts-trader rebalance --moc # market-on-close orders (see below)
msts-trader rebalance --order-type limit-chase # work each order as a limit pegged to the mid (see below)
msts-trader rebalance --min-weight 0.01 # ignore CSV rows under 1% weight
msts-trader rebalance --allocation 50000 # weights apply to $50k, not full NAV
msts-trader --broker paper rebalance --csv-file ... # test against paper
--moc(market-on-close): orders fill in the exchange closing auction instead of immediately — useful when your target weights are computed against closing prices. Supported on Alpaca, IBKR, Schwab, and paper (Tastytrade/Tradier/Hyperliquid have no MOC order type — the CLI refuses rather than silently downgrading). MOC orders are whole-share only, and exchanges stop accepting them around 15:50 ET, so submit before then. Also available asmoc = truein the config file.--order-type limit-chase: instead of one market order per leg, each order is worked as a LIMIT pegged to the live mid — re-quote and reprice every few seconds (--chase-interval, default 5s; polled every--chase-poll, default 1s) up to--chase-retriestimes (default 5), then fall back to a market order for whatever hasn't filled (disable with--no-chase-fallback).--chase-aggression 0.001nudges the limit 0.1% past the mid toward the fill side to improve the hit rate (default0= pure mid). The goal is execution quality — pay near the mid instead of crossing the whole spread. Safety: the prior limit is cancelled before each reprice (and the chase aborts rather than risk two live orders if a cancel fails), partial fills only re-submit the remainder, and no resting order is ever left behind. RTH only (the market fallback assumes the regular session), supported on all brokers; any that can't chase warn once and use market orders. Also available asorder_type = "limit-chase"in the config file (and in amulticonfig, including per-[[account]]override).--whole-shares: round every order down to whole shares (buys never exceed target, sells never exceed the held quantity). Use it for an IBKR account — or any broker/account — without fractional-trading permission on the API, which otherwise rejects fractional orders with error 10243 ("Fractional-sized order cannot be placed via API"). Applied before the preview, so what you see is exactly what's sent (and margin-aware scaling re-rounds to whole shares too). Also available aswhole_shares = truein the config file.--stop-pct: a default protective stop (fraction below entry, e.g.--stop-pct 0.015= 1.5%) applied to every bought/held target that has no per-rowstop_pct. An explicitstop_pctcolumn value always wins; exits (weight 0) get none. Use it when your weights feed carries only ticker+weight but you still want every position stopped. Alsostop_pctin the config file (and per-[[account]]in amulticonfig).--min-weight: rows with0 < weight < min-weightare ignored entirely — no buy, and an existing position in that ticker is not exit-swept either. An explicit weight of0still means "sell it all". Useful when the CSV carries many tiny weights you don't want to trade.--allocation: size the weights against a fixed dollar amount instead of the whole account — e.g. run a $50k strategy sleeve inside a $200k account. Positions in tickers not in the CSV are still exited (the sweep is account-wide), so keep sleeve and non-sleeve tickers disjoint or rebalance with a CSV that lists everything you hold. Capped at NAV; use leveraged weights (sum > 1.0) for gross exposure above the allocation.
Safety, automation & output flags
msts-trader rebalance --no-margin-aware # disable buying-power-fit scaling (on by default)
msts-trader rebalance --max-notional 60000 # refuse if gross buys exceed $60k
msts-trader rebalance --max-stale-hours 36 # refuse if the CSV's `# asof:` is too old
msts-trader rebalance --json # machine-readable output (one JSON object)
msts-trader rebalance --quiet # minimal output for cron logs
msts-trader rebalance --notify-url <webhook> # Discord/Slack/generic ping on execute
msts-trader rebalance --force # run even if same targets already done today
msts-trader rebalance --config my.toml # load defaults from a config file
- Idempotency: identical targets won't trade twice in the same UTC day
unless you pass
--force(guards against a cron + manual overlap). - Stale guard: add a
# asof: 2026-06-05T15:45:00Zcomment line to your CSV and--max-stale-hoursrefuses to trade on old weights. - Notifications: set
--notify-urlorMSTS_NOTIFY_URL(Discord/Slack/generic webhook), orMSTS_TELEGRAM_TOKEN+MSTS_TELEGRAM_CHAT_ID(Telegram creds can also go inconfig.tomlastelegram_token/telegram_chat_id). A failed webhook never blocks trading, but the failure is now reported (notify failed: webhook) instead of swallowed.--dry-runalso fires a clearly-labelled preview notification, so you can wire up and test a webhook without sending orders. - Retries: transient broker errors (429s, timeouts) are retried with backoff; real errors fail fast.
Config file
Set defaults once in ~/.msts-trader/config.toml (or pass --config):
broker = "tastytrade"
threshold = 0.04
csv_url = "https://example.com/weights.csv"
max_notional = 60000
max_stale_hours = 36
notify_url = "https://discord.com/api/webhooks/..."
telegram_token = "123456:ABC-DEF..." # optional, instead of MSTS_TELEGRAM_TOKEN
telegram_chat_id = "987654321" # optional, instead of MSTS_TELEGRAM_CHAT_ID
margin_aware = true # default; set false to disable buying-power-fit scaling
moc = false # set true to always use market-on-close orders
order_type = "market" # or "limit-chase": peg a limit to the mid, reprice, then market-fallback (RTH only)
chase_retries = 5 # limit-chase: reprice attempts before the market fallback
chase_interval = 5 # limit-chase: seconds to wait for a fill before repricing
chase_poll = 1 # limit-chase: status-poll cadence within each rung (seconds)
chase_aggression = 0 # limit-chase: fraction past the mid toward the fill side (0 = pure mid)
chase_fallback = true # limit-chase: market order for any unfilled remainder
whole_shares = false # set true to round every order to whole shares (IBKR/no-fractional accounts)
min_weight = 0.01 # ignore CSV rows with weight under 1%
stop_pct = 0.015 # default protective stop for rows with no per-row stop_pct (per-row wins)
allocation = 50000 # weights apply to $50k instead of full NAV
quiet = false
Resolution order for any setting: CLI flag > environment > config file > default.
Other commands
msts-trader status # NAV, positions, market status (default broker)
msts-trader status --json # machine-readable account snapshot (monitoring)
msts-trader status --creds-file x # headless status, no keychain
msts-trader doctor # health-check creds/connectivity/market for each broker
msts-trader doctor --broker ibkr # check one broker
msts-trader brokers # list supported + configured brokers
msts-trader logout --broker alpaca # clear stored creds for one broker
msts-trader paper-reset # reset paper book to starting cash
msts-trader --version
doctor is the fastest way to diagnose a broker: it shows, per broker,
whether credentials are present, whether it connects, your NAV, position
count, and a sample SPY quote — so permission/connectivity problems
(like the IBKR KID block) surface immediately.
What it does
- Parses your CSV into
{ticker: target_weight}. - Pulls live NAV, cash, buying power, and current positions from your broker.
- Quotes every relevant symbol via the broker's market-data API.
- Computes the dollar delta per ticker, skips anything within the drift threshold (default 4% of NAV).
- Sells tickers no longer in your targets.
- Sizes buys at the current quote, rounded to 2 decimals where the broker supports fractional MARKET orders.
- Shows the full plan and waits for
ybefore sending anything. - Submits MARKET DAY orders. Logs results to
~/.msts-trader/fills/.
Headless / automated (cron, GitHub Actions)
Everything works two ways:
- Manual:
msts-trader→ paste CSV → confirm withy. - Headless: drive it entirely from files / env vars + flags — no
paste, no confirm prompt, no interactive
login, no keychain.
The headless one-liner:
msts-trader rebalance \
--broker tastytrade \
--creds-file creds.json \
--csv-url https://example.com/your-weights.csv \
--yes
--creds-file— JSON orKEY=VALUEfile with your credentials (or just export the env vars; both work). Seeexamples/creds.example.json.--csv-file PATHor--csv-url URL— the target weights, instead of pasting.--yes— skip the confirmation prompt (required for unattended runs).--dry-run— preview only, never sends (great for a first test).
Credentials resolve in this order: --creds-file / environment first,
then the OS keychain. So a server or CI box that has never run login
works as long as the env vars are set.
Ready-to-use templates are in examples/:
rebalance-cron.sh— a cron wrapper.github-action-rebalance.yml— a scheduled GitHub Actions workflow.
Broker notes for automation:
- Tastytrade, Alpaca, and Tradier are pure REST/OAuth → work in GitHub Actions or any server.
- IBKR needs a running TWS / IB Gateway on a machine you control → use cron on that machine, not GitHub Actions.
The market-hours guard still applies: a headless run outside US regular hours exits without trading, so a daily schedule is safe.
Exit codes
For scripting, rebalance / multi use:
| Code | Meaning |
|---|---|
0 |
Success — executed, or nothing to do (within drift / dry-run / duplicate) |
1 |
Error — bad/missing creds, malformed CSV, a blocker (e.g. --max-notional), stale CSV, or a partial/failed execution |
2 |
Market closed or not in a regular-hours session (equities) |
Multiple accounts
Run the same target weights across several accounts in one pass with the
multi command and a TOML config that lists each account's broker and
creds file:
# multi-account.toml
csv_url = "https://example.com/weights.csv"
threshold = 0.04
max_notional = 60000
[[account]]
name = "tasty-main"
broker = "tastytrade"
creds_file = "~/.msts-trader/tasty.json"
[[account]]
name = "alpaca-live"
broker = "alpaca"
creds_file = "~/.msts-trader/alpaca.json"
msts-trader multi --config multi-account.toml --dry-run # preview all
msts-trader multi --config multi-account.toml --yes # execute all
msts-trader multi --config multi-account.toml --json --yes # machine-readable
Accounts run sequentially; each gets its own credentials (no cross-leak),
the same idempotency + safety checks as a single run, and a combined
summary at the end. multi never prompts — --yes is required to
execute, --dry-run to preview. See
examples/multi-account.toml.
Protective stops
Add an optional stop_pct column to the CSV and msts-trader places a
GTC SELL STOP under each position it buys:
ticker,weight,stop_pct
SPY,0.42,
GLD,0.18,0.05
WGMI,0.02,0.015
stop_pctis a fraction below the fill price, not a price:0.05= 5%,0.015= 1.5%. Must be in(0, 0.5); a blank cell means no stop.- After a BUY fills, a GTC SELL STOP is placed for the filled quantity at
fill_price × (1 − stop_pct). - Stops are reconciled every rebalance: on a SELL the existing stop is cancelled (and re-placed on the remaining quantity if you still hold some and the target still wants a stop), so a resting stop never outlives its position and turns into a naked short.
- Supported on 6 of 7 brokers — Tastytrade, Alpaca, Tradier, IBKR, Schwab, and paper. Hyperliquid has no stop support: the column is ignored with a one-time warning, weights still execute. Verify a broker honors stops with a 1-share test before relying on it.
See examples/pnl-unified.toml for a full
copy-trade + stop setup.
Leveraged weights
Target weights are fractions of your account NAV. They can sum to more
than 1.0 — that's leverage. For example a book that sums to 1.60
(160% gross exposure, 1.60x) sizes each position at weight × NAV, and
the amount over 100% is financed on margin:
ticker,weight
QQQ,0.3123
GLD,0.2537
TBT,0.1480
... # sums to ~1.60 = 160% gross
The preview shows Gross target exposure: 160% (1.60x). Margin-aware
sizing is on by default (matching a production live runner): if the
buys exceed your available buying power (broker BP plus the proceeds from
the sells, which execute first), msts-trader scales every buy by one
uniform factor so the whole book fits — preserving your relative
weights — instead of letting the broker reject the tail of the order set
piecemeal and distort your allocation. When the sells already fund the
buys, nothing is scaled (and it's free — a notional pre-check skips the
broker margin queries unless the book is actually tight). Pass
--no-margin-aware to disable.
Where the broker exposes it, this uses the broker's real per-order margin so leveraged-ETF rates (TBT, EDZ, …) are sized exactly — the same approach a production live runner uses:
| Broker | Margin source |
|---|---|
| Tastytrade | real — order dry-run buying_power_effect |
| IBKR | real — whatIfOrder initial-margin change |
| Tradier | real — order preview margin_change |
| Alpaca / Schwab | buying power (already encodes the Reg-T 2× multiplier) |
Real per-order margin only matters for leveraged ETFs; for plain ETFs, notional-vs-buying-power is already exact. All paths are weight-preserving, and any failure to get real margin falls back to the notional estimate automatically (never sizes on partial data).
With real margin it also re-confirms: after scaling, it re-queries the broker on the now-smaller book and scales again if non-linear margin tiers still push it over (up to a few passes), then reports one cumulative scale. The notional path is linear, so it's exact in a single pass.
Orders always execute sells before buys, so proceeds free up buying power before the buys submit (required on cash accounts, lower peak margin on margin accounts).
Two things to know for a fresh account:
- Positions smaller than the drift threshold (default 4% of NAV)
won't be established on the first run — they look "within drift" of a
zero holding. For initial setup of a book with small sleeves, lower it:
msts-trader rebalance --threshold 0.01. - A single weight above 3.0 (300%) is rejected as a likely
percentage-paste mistake (e.g.
31.23instead of0.3123).
What it does NOT do (yet)
- Pre-market or after-hours execution for equities. Refuses outside 09:30–16:00 ET (crypto via Hyperliquid trades 24/7).
- Shorting. Negative weights are rejected.
- Options or futures.
- Active stop management (Hydra/Fusion-style trailing watchers). Static
protective stops are supported via the
stop_pctCSV column — see Protective stops. - Scheduling itself (use cron / GitHub Actions — see Headless).
Troubleshooting
Can't paste or type during msts-trader login?
Some terminals — VS Code, Cursor, and Windows Terminal / Windows
consoles — don't reliably forward input to hidden-password prompts
(Python's getpass). The cursor sits there and nothing registers.
msts-trader detects these terminals and switches to visible input
automatically (you'll see a [notice]), so you can paste your secret —
it's just shown on screen as you type. But the cleanest fix is to not
type secrets at all:
Best: use a credentials file (--creds-file)
Create a small file — JSON or KEY=VALUE — with your credentials:
tt_creds.json
{
"TT_PROVIDER_SECRET": "your-provider-secret",
"TT_REFRESH_TOKEN": "your-refresh-token",
"TT_ACCOUNT_ID": "your-account-number"
}
or tt_creds.env
TT_PROVIDER_SECRET=your-provider-secret
TT_REFRESH_TOKEN=your-refresh-token
TT_ACCOUNT_ID=your-account-number
then:
msts-trader login --broker tastytrade --creds-file tt_creds.json
No prompts, no terminal quirks, works identically on every OS. Delete the file afterwards — the credentials are now in your OS keychain.
Lowercase keys (provider_secret, api_key, etc.) also work, and
client_secret is accepted as an alias for the provider secret (it's
what Tastytrade's portal calls it). Add TT_TEST=1 if the keys are from
Tastytrade's certification (sandbox) environment. For
Alpaca use APCA_API_KEY_ID / APCA_API_SECRET_KEY / APCA_PAPER;
for IBKR IBKR_HOST / IBKR_PORT / IBKR_CLIENT_ID /
IBKR_ACCOUNT_ID (optional — auto-discovered when omitted); for Schwab
SCHWAB_APP_KEY / SCHWAB_APP_SECRET / SCHWAB_CALLBACK_URL
(optional — defaults to https://127.0.0.1:8182; must exactly match
your app's registered callback, trailing slash included).
Or: set environment variables
Mind the shell — this trips people up:
- macOS / Linux (bash/zsh):
export TT_PROVIDER_SECRET="..." export TT_REFRESH_TOKEN="..." export TT_ACCOUNT_ID="..."
- Windows PowerShell (the Windows Terminal default —
exportandsetdo NOT work here):$env:TT_PROVIDER_SECRET="..." $env:TT_REFRESH_TOKEN="..." $env:TT_ACCOUNT_ID="..."
- Windows cmd.exe (do NOT wrap values in quotes — cmd keeps them):
set TT_PROVIDER_SECRET=... set TT_REFRESH_TOKEN=... set TT_ACCOUNT_ID=...
Then run msts-trader login --broker tastytrade in the same window.
(msts-trader strips accidental surrounding quotes, but PowerShell vs cmd
syntax still matters.)
login failed: invalid_grant / Grant revoked
This is Tastytrade telling you the refresh token is no longer valid — it was regenerated, the OAuth grant was revoked, or it expired from inactivity. It is not a bug in msts-trader; the token simply needs to be re-minted:
- https://developer.tastytrade.com → My Apps → your app
- Run the OAuth authorization flow again to get a new refresh token
msts-trader login --broker tastytrade(or--creds-file) with the new token
You'll also see this error if you use certification (sandbox) keys
against production — cert keys only work with TT_TEST=1 set.
Security
- Your broker credentials live only in your OS keychain on your own machine. The app does not phone home, does not log credentials, and is not connected to any service operated by the author.
- The author of this app cannot view, recover, or revoke your broker access. Revoke via your own broker's API-app dashboard if a key leaks.
- Trades are user-initiated: every execution requires you to paste a
CSV and confirm with
y. There is no background trading loop.
Full details and how to report a vulnerability: SECURITY.md.
Disclaimer
This tool sends real orders to your live brokerage account. You are responsible for the CSV you paste and the rebalance you confirm. Past performance of any signal source is not indicative of future results. The author makes no warranty of any kind; use at your own risk.
Changelog
See CHANGELOG.md for the full version history. Each released tag also has a GitHub Release with the same notes and the built wheel attached.
Development
git clone https://github.com/markudevelop/msts-trader.git
cd msts-trader
pip install -e ".[all,dev]"
pytest -v # 350+ tests, a couple of seconds
ruff check msts_trader
or with uv (uses the Python pinned in .python-version):
uv sync --all-extras
uv run pytest -v
uv run ruff check msts_trader
The test suite covers:
- CSV parser (header validation, weights, leverage, comments, dup/neg guards)
- Diff math (drift threshold, exits, warnings, blockers, BP overrun, leverage)
- Market hours (RTH/pre/after/closed, holidays through 2027, weekends)
- Paper broker end-to-end (cash accounting, position lifecycle, dry-run, persistence)
- Broker protocol conformance (every adapter exposes the required attrs + methods)
- Keychain + env-derived credentials (per-broker, quote stripping, fallbacks)
- Safety (max-notional cap, stale-CSV guard), retry/backoff, idempotency
- Config file parsing, notifications formatting/dispatch
- CLI (help, version, brokers list, doctor, login, no-creds clean exit)
Live brokerage adapters are not exercised against real APIs in CI — they need credentials and can move real money. The tests verify structure; you verify fills.
License
PolyForm Noncommercial License 1.0.0.
You may use, modify, and share this software for any noncommercial purpose — personal trading, research, education, hobby projects. Selling, hosting as a paid service, or otherwise commercializing this software or derivative works is not permitted without a separate commercial license. Contact the author if you need one.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file msts_trader-0.22.0.tar.gz.
File metadata
- Download URL: msts_trader-0.22.0.tar.gz
- Upload date:
- Size: 156.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
a023b847d340c44eb226b58784dfc24a24734134751e2e511b312ba59d45f29c
|
|
| MD5 |
6efc001dafe58fde657c36ef26baef8a
|
|
| BLAKE2b-256 |
9c1850c31529c6a440158891ad8992ad897b4a95fcde67c9b19f5a160ae3cdd8
|
Provenance
The following attestation bundles were made for msts_trader-0.22.0.tar.gz:
Publisher:
release.yml on markudevelop/msts-trader
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
msts_trader-0.22.0.tar.gz -
Subject digest:
a023b847d340c44eb226b58784dfc24a24734134751e2e511b312ba59d45f29c - Sigstore transparency entry: 1870370411
- Sigstore integration time:
-
Permalink:
markudevelop/msts-trader@8799031ebe02a05e459363e55ae5abca7de8eaec -
Branch / Tag:
refs/tags/v0.22.0 - Owner: https://github.com/markudevelop
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@8799031ebe02a05e459363e55ae5abca7de8eaec -
Trigger Event:
push
-
Statement type:
File details
Details for the file msts_trader-0.22.0-py3-none-any.whl.
File metadata
- Download URL: msts_trader-0.22.0-py3-none-any.whl
- Upload date:
- Size: 100.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6b3ed83cacfce209dcc907e04972af261f24ed95605184e0d2af869b00d9a721
|
|
| MD5 |
74046a6c3bf6a8393f9f6a35da3d7557
|
|
| BLAKE2b-256 |
901898d4cbc394e12f02c28e49bffd8143837416732e19bc1a876cfc7b7cfe47
|
Provenance
The following attestation bundles were made for msts_trader-0.22.0-py3-none-any.whl:
Publisher:
release.yml on markudevelop/msts-trader
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
msts_trader-0.22.0-py3-none-any.whl -
Subject digest:
6b3ed83cacfce209dcc907e04972af261f24ed95605184e0d2af869b00d9a721 - Sigstore transparency entry: 1870370447
- Sigstore integration time:
-
Permalink:
markudevelop/msts-trader@8799031ebe02a05e459363e55ae5abca7de8eaec -
Branch / Tag:
refs/tags/v0.22.0 - Owner: https://github.com/markudevelop
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@8799031ebe02a05e459363e55ae5abca7de8eaec -
Trigger Event:
push
-
Statement type: