Market making is a trading strategy where the trader simultaneously places both buy and sell orders in an attempt to profit from the bid-ask spread. Market makers stand ready to both buy and sell from other traders, thus providing liquidity to the market.
The strategy is appealing to traders because it doesn't require traders to take a directional view of the market - there's money to be made when the market goes up and when the market goes down. It's also heavily incentivized by exchanges looking for liquidity and volume - many exchange operators will pay you to make markets on their exchanges.
Let's consider a simplified market. Let's say there are three traders: Alice, Bob, and Tim. Alice is looking to sell some of her Bitcoins and Bob is looking to convert some of his USD into Bitcoin. Neither one are savvy about markets or cryptocurrency - they use Bitcoin, but aren't going to lose sleep over trying to get the absolute best prices. Tim is operating Krypto-trading-bot. Now lets say that the price BTC to USD is $100. Tim could configure Krypto-trading-bot to send in a buy order for 1 BTC at $95 and a sell order for 1 BTC at $105. Tim would hope that Bob would come along and buy the offered sell order at $105 and Alice would come and sell BTC at $95 - netting Tim $10.
But what if that doesn't happen? What if the price of BTC/USD jumps to $103? Now Tim's buy order seems really uncompetitive at $95 - Alice doesn't want to sell for that little. And Bob could get a pretty good deal by getting the BTC at only two extra dollars. To prevent this scenario, Tim's Krypto-trading-bot would readjust the orders by cancelling the $95-$105 orders and placing a new set of orders - also known as making a market - at $98-$108.
Like in any market these days, Bob and Alice probably aren't humans clicking buttons. They certainly aren't humans shouting on a floor in lower Manhattan. More likely, they are also computerized algorithms capable of placing orders in milliseconds. To survive as a market maker, you need to be faster than those algorithms to make a profit.
As previously mentioned, market making is really the art of figuring out the price of something, then making a market around that price. So how do we know what is the real price of Bitcoin? Well... we don't. And of course the price of Bitcoin now might be radically different than the price in a day, in an hour, or even in a second from now. The best we can do is to build an estimate, or fair value, of the price of Bitcoin. In Krypto-trading-bot, we consume the market data from the exchange we are sending orders into as a starting point. That includes the best bids and offers and most recent trades (aka market trades) by other participants in the market.
From our fair value, we then need to make a market around that price. Back to our hypothetical example with Bob and Alice: we could make our market as $95-$105, we could have also made it $99.99-$101.01, or we could have also done $47-$332.21. So how do we decide? That's where the quoting parameters come into play. Those parameters dictate how wide of a market to make (wide=askPrice-bidPrice) and what size we want our quotes in the market to be. The procedure for coming up with profitable parameters is both an art and a science - there is no one size fits all formula.
When Krypto-trading-bot figures out a suitable market, Krypto-trading-bot will then send in the buy and sell orders. Hopefully it's able to buy for less then sell for more, and repeat many times per day. Sometimes that's not always the case - sometimes there are genuinely more buyers than sellers for the prices you are setting. Often this comes when the market is moving very fast in one direction. Luckily, Krypto-trading-bot will prevent you from selling too fast without finding corresponding buyers and will stop sending orders in the imbalanced side.
The code is organized into 3 layers.
-
Engine layer - The brains of the application. Portion of the code responsible for synthesizing market data, open order status, position, fees, trades, safety information and converting that into a quote to send to the exchange. This is the portion of the code which calculates a fair value (
FairValueEngine
) and generates quotes (QuotingEngine
) -
Adapter layer - The engine layer should have no idea about the individual quirks of the exchanges. The engine layer uses the adapter layer to carry out its bidding. The adapter layer also has no idea that it is being used in a manner to make markets. In theory, the adapter layer code and the gateway layer code could be divorced from the Engine layer and we could build a technical analysis or latency arbitrage bot, instead. The adapter layer also contains all the state reported by the gateways.
-
Gateway layer - Each exchange has their own API for interacting with the exchange. All of that business is hidden behind 4 different interfaces:
IMarketDataGateway
: Handles order book updates and market trade updates.IOrderEntryGateway
: Send and cancel orders and handle updates to the orders.IPositionGateway
: Pulls in the latest position (how much BTC and USD do I have?) informationIExchangeDetailsGateway
: Read-only information describing naming and exchange fee structure.
Gateways are ideally stateless (some state may be needed in order to perform exchange-specific functionality) and are mostly translation layers between exchange APIs and the Krypto-trading-bot API.
Navigate to the Web UI as described in the install process. You should see a screen like:
-
Market Data and Quotes - this is perhaps the most important screen in the app.
-
The top row in blue with "q" is the quote that you generate via the Quoting Parameters supplied. If the text is grey, the generated quotes are not in the market, and green when they are.
-
"FV" is the fair value price calculated by Krypto-trading-bot as the starting point for the generated quote.
-
The "mkt" rows are the best bids and offers on the exchange you are connected to.
-
-
Trades - Trades done by your exchange account. "side" is the side which your order was sent as. "val" is the total value of the trade, which is price * size +/- total exchange fee
-
Market Trades - Trades done by all participants in the market. "ms" is the side that was the make side (provided liquidity). The columns starting with "q" are your quotes at the time of the trade, the columns starting with "m" are the best bid and offer information at the time of the trade.
-
Quoting Parameters - All of the parameters needed to generate a quote. See the section "How do I control Krypto-trading-bot' quotes?" for each field's description.
-
Positions - Shows holdings of each currency pair you are using.
-
Order List - Shows order statuses of each order sent to the exchange.
- "Cxl" - clicking the red button will attempt to cancel the order.
In the web UI, there are three rows of panels with cryptic looking names and editable textboxes. Those are the quoting parameters, the knobs which we can turn to affect how Krypto-trading-bot will trade.
-
%
- If enabled, the values ofbidSize
,askSize
,tbp
,pDiv
andrange
will be a percentage related to the total funds (available + held in both sides); useful when the very same funds are used in multiple markets, so the quantity of the funds is highly variable, then may be useful to work with percentages. -
mode
- Sets the quoting mode-
Join
- Sets our quote to be at the best bid and the best offered price, If the BBO is narrower thanwidth
, set our bid quote atFV - width / 2
and ask quote atFV + width / 2
. -
Top
- Same asJoin
, but if the code can better the best bid or offer by a penny while respecting thewidth
, set that as the quote so we will then be at the top of the market. -
Mid
- Set our bid quote atFV - width / 2
and ask quote atFV + width / 2
-
Inverse Join
- Set the quote at the BBO if the BBO is narrower thanwidth
, otherwise make the quote so wide that no one will trade with it. -
Inverse Top
- Same asInverse Join
but make our orders jump to the very top of the order book. -
HamelinRat
- Follow the Colossus of the market. Unlike other modes, it does not calculate the quote spread based on fair value, instead it looks for the biggest order in the market levels and places the quote right before it. -
Depth
- Usewidth
asdepth
. Unlike other modes, it does not calculate the quote spread based on fair value, instead it walks over all current open orders in the book and places the quote right afterdepth
quantity, at both sides.
-
-
safety
- Sets a quoting Safety-
PingPong
- Always respect the calculatedwidthPong
from the last sold or boughtsize
, if any. -
Boomerang
- Same asPingPong
but the calculatedwidthPong
for new Pongs is based on any best matching previous sold or boughtsize
, if any. -
AK-47
- Same asBoomerang
but allows multiple orders at the same time in both sides. To avoid old trades, on every new trade Krypto-trading-bot will cancel all previous trades if those are worst.
-
-
bullets
- Maximum amount of trades placed in each side (only affectsAK-47
). -
range
- Minimum width betweenbullets
in USD (ex. a value of .3 is 30 cents; only affectsAK-47
). -
pingAt
(Pongs are always placed in both sides, only affectsPingPong
,Boomerang
andAK-47
)-
BothSides
- Place new Pings in both sides. -
BidSide
- Place new Pings only in Bid side, and therefore in Ask side only Pongs will be placed. -
AskSide
- Place new Pings only in Ask side, and therefore in Bid side only Pongs will be placed. -
DepletedSide
- Place new Pings only in the opposite side with not enough funds to continue trading. -
DepletedBidSide
- Place new Pings only in the Ask side if there are not enough funds to continue trading in the Bid side. -
DepletedAskSide
- Place new Pings only in the Bid side if there are not enough funds to continue trading in the Ask side. -
StopPings
- Only place new Pongs based on the history of Pings, without placing new Pings.
-
-
pongAt
(only affectsPingPong
,Boomerang
andAK-47
)-
ShortPingFair
- Place new Pongs based on the lowest margin Ping in history respecting thewidthPong
from thefair value
. -
LongPingFair
- Place new Pongs based on the highest margin Ping in history respecting thewidthPong
from thefair value
. -
ShortPingAggresive
- Place new Pongs based on the lowest margin Ping in history without respecting thewidthPong
from thefair value
. -
LongPingAggresive
- Place new Pongs based on the highest margin Ping in history without respecting thewidthPong
from thefair value
.
-
-
bw?
- Enable Best Width to place orders avoiding "hollows" in the book, while accomodating new orders right near to existent orders in the book, without leaving "hollows" in between. -
%w?
- If enabled, the values ofwidth
orwidthPing
andwidthPong
will be a percentage related to thefair value
; useful when calculating profits subtracting exchange's fees (that usually are percentages too). -
width
andwidthPing
- Minimum width (spread) of our quote in USD (ex. a value of .3 is 30 cents). With the exception for whenapr
is checked and the system is aggressively rebalancing positions after they get out of whack,width
will always be respected. -
widthPong
- Minimum width (spread) of our quote in USD (ex. a value of .3 is 30 cents). Used only if previous Pings exists in the opposite side. -
bidSize
- Maximum bid size of our quote in BTC (ex. a value of 1.5 is 1.5 bitcoins). With the exception for whenapr
is checked and the system is aggressively rebalancing positions after they get out of whack. -
askSize
- Maximum ask size of our quote in BTC (ex. a value of 1.5 is 1.5 bitcoins). With the exception for whenapr
is checked and the system is aggressively rebalancing positions after they get out of whack. -
maxBidSize?
andmaxAskSize?
- UsebidSize
andaskSize
as minimums and automatically find the maximum possiblesize
based on the current "Target Base Position" (just as having enabledapr
onSize
but even before your position diverges more thanpDiv
). -
fv
- Sets the fair value calculation mode-
BBO
-FV = ([best bid price] + [best ask price])/2.0
-
wBBO
-FV = ([best bid price]*[best ask size] + [best ask price]*[best bid size])/([best ask size] + [best bid size])
-
-
apMode
-
Manual
- Krypto-trading-bot will not try to automatically manage positions, instead you will need to manually settbp
. -
EWMA_LS
- Krypto-trading-bot will use along
minute andshort
minute exponential weighted moving average calculation to buy up BTC when theshort
minute line crosses over thelong
minute line, and sell BTC when the reverse happens. The EWMA values are currently exposed in the stats. -
EWMA_LMS
- Krypto-trading-bot will use along
minute,medium
minute andshort
minute exponential weighted moving average calculation, together with the simple moving average of the last 3fair value
values, to buy up BTC when theshort
minute line crosses over thelong
minute line, and sell BTC when the reverse happens. -
EWMA_4
- Krypto-trading-bot will use amedium
minute andsmall
minute EWMA calculation to buy when thesmall
minute line crosses over themedium
minute line, and sell when the reverse happens. Additionally sets thetbp
to 0% if theverylong
EWMA minute line crosses over thelong
EWMA minute line.
-
-
verylong
- Only used whenapMode
isEWMA_4
. Sets the periods of EWMA VeryLong to automatically manage positions. -
long
- Only used whenapMode
isEWMA
. Sets the periods of EWMA Long to automatically manage positions. -
medium
- Only used whenapMode
isEWMA_LMS
orEWMA_4
. Sets the periods of EWMA Medium to automatically manage positions. -
short
- Only used whenapMode
isEWMA
. Sets the periods of EWMA Short to automatically manage positions. -
sensibility
- Threshold removed from each period, affects EWMA Long, Medium and Short. The decimal value ofsensibility
must be betweem 0 and 1. -
tbp
- Only used whenapMode
isManual
. Sets a static "Target Base Position" for Krypto-trading-bot to stay near. In manual position mode, Krypto-trading-bot will still try to respectpDiv
and not make your position fluctuate by more than that value. So if you have 10 BTC to trade, settbp = 3
, setapMode = Manual
, andpDiv = 1
, your holding of BTC will never be less than 2 or greater than 4. -
pDivMode
- Only used whenapMode
is notManual
mode. Sets the strategy of dynamically adjusting thepDiv
depending on the divergence from 50% of Base Value.-
Manual
- No dynamic adjusting ofpDiv
. -
Linear
- Linear calculation betweenpDiv
andpDivMin
. -
Sine
- Calculation betweenpDiv
andpDivMin
on a sine curve. -
SQRT
- Square root calculation betweenpDiv
andpDivMin
. -
Switch
- Iftbp
is more than 90% or less than 10%,pDivMin
is taken, otherwhisepDiv
.
-
-
pDiv
- If your "Target Base Position" diverges more from this value, Krypto-trading-bot will stop sending orders to stop too much directional trading. So if you have 10 BTC to trade, "Target Base Position" is reporting 5, andpDiv
is set to 3, your holding of BTC will never be less than 2 or greater than 8. -
pDivMin
- Only used whenpDivMode
is notManual
. It defines the minimalpDiv
for the dynamic positon divergence. -
apr
- If you're in a state where Krypto-trading-bot has stopped sending orders because your position has diverged too far from Target Base Position, this setting will much more aggressively try to fix that discrepancy by placing orders much larger thansize
and at prices much more aggressive thanwidth
normally allows (seepongAt
option). It's a bit risky to use this setting.-
Off
- Krypto-trading-bot will not try to aggressively try to stabilize the target based position. -
Size
- Krypto-trading-bot will aggressively make use of biggersize
values (x3size
or half of the diverged target base position, whatever is smaller). -
SizeWidth
- Same asSize
but also will aggressively make use of smallerwidth
values (respecting always aggressivepongAt
option andwidthPong
).
-
-
aprFactor
- Defines the value with which thesize
is multiplicated whenapr
is in functional state. -
sop
- Super opportunities, if enabled and if the market width issopWidth
times bigger than thewidth
set, it multipliessopTrades
totrades
and/orsopSize
tosize
, in both sides at the same time. -
sopWidth
- Is the value with the market width is multiplicated to define the activation point for Super opportunities. -
sopTrades
- Multiplicatestrades
to rise the possible Trades per Minute ifsop
is inTrades
ortradesSize
state. -
sopSize
- Multiplicateswidth
ifsop
is inSize
ortradesSize
state. -
trades
- Often, only buying or selling many times in a short timeframe indicates that there is going to be a price swing.trades
and/sec
are highly related: If you successfully complete more orders thantrades
in/sec
seconds, Krypto-trading-bot will stop sending more buy orders until either/sec
seconds has passed, or you have sold enough at a higher cost to make all those buy orders profitable. The number of trades is reported by side in the UI; "BuyTS", "SellTS", and "TotTS". If "BuyTS" goes abovetrades
, Krypto-trading-bot will stop sending buy orders, and the same for sells. For example, iftrades
is 2 and/sec
is 1800 (half an hour):
Time | Side | Price | Size | BuyTS | SellTS | Notes |
---|---|---|---|---|---|---|
12:00:01 | Buy | 10 | 1 | 1 | 0 | |
12:00:02 | Buy | 10 | 0.5 | 1.5 | 0 | Partial fills of size get counted fractionally |
12:00:03 | Sell | 11 | 0.75 | 0.75 | 0 | Sell for more decrements the imbalance |
12:00:05 | Sell | 5 | 0.75 | 0.75 | 0 | Sell for less than the other buys doesn't help |
12:00:06 | Buy | 10 | 0.5 | 1.75 | 0 | |
12:00:07 | Buy | 10 | 0.5 | 2.75 | 0 | Stop sending buy orders until 12:30:07! |
-
/sec
- seetrades
. -
ewmaPrice?
- Use a quote protection ofperiods
smoothed line of the fair value to limit the price while sending new orders. -
ewmaWidth?
- Use a quote protection ofperiods
smoothed line of the width (between the top bid and the top ask) to limit the widthPing while sending new orders. -
periodsᵉʷᵐᵃ
- Maximum amount of values collected in the sequences used to calculate theewmaPrice?
andewmaWidth?
quote protection. After collect sequentially every 1 minute the value of thefair value
, and before place new orders, a limit will be always applied to the new orders price using aewma
calculation, taking into account only the lastperiods
periods in each sequence. -
ewmaTrend?
- Use a trend protection of doubleperiods
(Ultra+Micro) smoothed lines of the price to limit uptrend sells and downtrend buys. -
threshold
- When trend stregth is above positive threshold value bot stops selling, when strength below negative threshold value bot stops buying. -
ultra
- Time in minutes to define Ultra EMA -
micro
- Time in minutes to define Micro EMA -
stdev
-
Off
- Do not limit the price of new orders. -
OnFV
- Use a quote protection of STDEV, calculated from a sequence offair value
values duringperiods
periods of 1 second, to limit the price equally on both sides while sending new orders. -
OnFVAPROff
- Same asOnFV
whenapr
isOff
or when the system is not aggressively rebalancing positions; otherwise if is rebalancing, is same asOff
. -
OnTops
- Use a quote protection STDEV, calculated from a unique sequence of bothbest bid
andbest ask
values in the market order book duringperiods * 2
periods of 1 second, to limit the price equally on both sides while sending new orders. -
OnTopsAPROff
- Same asOnTops
whenapr
isOff
or when the system is not aggressively rebalancing positions; otherwise if one side is rebalancing, is same asOff
for that side. -
OnTop
- Use a quote protection STDEV, calculated from two sequences of thebest bid
(first sequence) and also of thebest ask
(second sequence) value in the market order book duringperiods
periods of 1 second, to limit the price independently on each side while sending new orders. -
OnTopAPROff
- Same asOnTop
whenapr
isOff
or when the system is not aggressively rebalancing positions; otherwise if one side is rebalancing, is same asOff
for that side.
-
-
periodsˢᵗᵈᶜᵛ
- Maximum amount of values collected in the sequences used to calculate the STDEV, each side may have its own STDEV calculation with the same amount ofperiods
. After collect sequentially every 1 second the values of thefair value
,last bid
and also of thelast ask
from the market order book, and before place new orders, a limit will be always applied to the new orders price using a calculation of the STDEV, taking into account only the lastperiods
periods in each sequence. -
factor
- Multiplier used to increase or decrease the value of the selectedstdev
calculation, afactor
of 1 does effectively nothing. -
BB?
- Enable Bollinger Bands with upper and lower bands calculated from the result of the selectedstdev
above or below its own moving average ofperiods
. -
cxl?
- Enable a timeout of 5 minutes to cancel all orders that exist as open in the exchange (in case you found yourself with zombie orders in the exchange, because the API integration have bugs or because the connection is interrupted). -
profit
- Timeframe in hours to calculate the display of Profit (under wallet values) and also interval in hour to remove data points from the Stats, for example aprofit
of 0.5 will compare the current wallet values and the values from half hour ago to display the +/- % of increment between both and will remove data from the Stats older than half an hour. -
Kmemory
- Timeout in days for Pings (yet unmatched trades) and/or Pongs (K trades) to remain in memory, a value of0
keeps the history in memory forever; a positive value remove only Pongs afterKmemory
days; but a negative value remove both Pings and Pongs afterKmemory
days (for example a value of-2
will keep a history of trades no longer than 2 days without matter if Pings are not matched by Pongs; or a value of-0.25
will do so but limited to 6h). -
delayUI
- Relax the display of UI data bydelayUI
seconds. Set a value of 0 (zero) to display UI data in realtime, but this may penalize the communication with the exchange if you end up sending too much frequent UI data (like in low latency environments with super fast market data updates; at home is OK in realtime because the latency of Krypto-trading-bot with the exchange tends to be higher than the latency of Krypto-trading-bot with your browser). -
audio?
- plays a sound for each new trade (ping-pong modes have 2 sounds for each type of trade).
You can always open a new issue to request a new feature.