/martin-binance

Free trading system for Binance SPOT market. Adaptive customizable reverse grid strategy based on martingale. Binance API multiplexed wrapper for strategy developer.

Primary LanguagePythonMIT LicenseMIT


Modified Martingale

Cyclic grid strategy for SPOT market

Free trading system for Binance API


PyPI version DeepSource DeepSource Total alerts Language grade: Python

Many other crypto exchanges available through multi-exchange terminal margin.de

The motto of the project

Profitable, fault-tolerant, adaptable to the market. Started and forgot. Don't forget to pick up what you earned.

Regardless of any trend, exchange overloads, network connection lost, hardware fault.

Disclaimer

All risks and possible losses associated with use of this strategy lie with you. Strongly recommended that you test the strategy in the demo mode before using real bidding.

Review

The system can be used in two modes:

  • STANDALONE, for free unlimited trading on Binance SPOT market.
  • python_strategy modules can be used as plug-in trading strategy for multi-exchange terminal margin.de Free demo, you can try it.

Strategy logic at executor.py file and trading parameters set in the API_1_BTCBUSD.py (cli_7_BTCUSDT.py)

You can modify them for your needs. See For developers section.

Reference

Trade idea

Features

Quick start

Terminal Tmux for STANDALONE mode

How it's work

For developers

Known issue

Target

Referral code and donat

Trade idea

Create a grid of increasing volume orders and when they perform creation of one take profit order in the opposite direction.

Its volume is equal to the sum of the executed grid orders, and the price compensates the fee and provides the specified profit.

What is the chip? After each grid order executed, the price of the take profit order (TP) approaches the price of the grid, which requires less bounce to perform it.

If all grid orders filled, then reverse and start the cycle in another direction. The price overlap set for the reverse cycle grid provides a given profit for the initiating cycle.

For Reverse cycle there are only two possible result depending on the price in the market. In the first case the entire grid executed therefore we close the previous cycle with profit. In second case executing part of the grid orders, and the filling take profit order increase the depo in the second coin. It reduces the necessary price overlap and sooner or later first variant comes true.

This allows you to increase the initial deposit using price fluctuations in any trend, and makes this strategy really break even. Unless you trade scam shitcoin.

In the cycle to sell, the profit accumulates in the first currency. In the cycle to buy, the profit accumulates in the second coin.

Fee payments are taken into account when calculating TP.

The optimal pair choice is a stablecoin or fiat plus a coin from the top ten.

Features

  • Create grid and take profit orders
  • Logarithm price option for grid orders (customizable)
  • Reverse algo if all grid orders are filling
  • Calculation of the price overlap for the Reverse cycle for the profitable completion of the previous cycle
  • Use Average Directional Index indicator for best time starting Reverse cycle
  • Calculation of separate MAKER and TAKER fee
  • Shift grid orders (customizable) if the price is go way (before take profit placed)
  • Fractional creation of grid for increase productivity (customizable)
  • Adaptive overlap price for grid on current market conditions (customizable) based on Bollinger Band
  • Adaptive profit setting on current market conditions (customizable) based on Bollinger Band
  • Adaptive grid orders quantity for initial and Reverse cycle
  • Stable operation in pump/dump conditions
  • Process partial execution of orders with large amounts of funds
  • Save funding change, cycle parameter and result in sqlite3 .db for external analytics
  • Telegram notification
  • External control from Telegram bot
  • Restore state after restart strategy

Quick start

Install

pip install martin-binance

Create Telegram bot

  • Register Telegram bot
  • Get token
  • Find channel_id. Just start IDBot and get channel_id
  • Specify this data into ms_cfg.toml for 'Demo - Binance', 7

STANDALONE mode

  • Log in at Binance Spot Test Network
  • Create API Key
  • Specify api_key and api_secret in binance_srv_cfg.toml for 'Demo - Binance'
  • Run binance_srv.py in terminal window
  • Run cli_7_BTCUSDT.py in other terminal window

Strategy is started.

Setting trade pair. You must set pair name in three places the same (yes, it is crooked, but so far):

  • base setting at bottom of the cli_X_AAABBB.py in "main" section, SYMBOL = 'AAABBB'
  • the name of cli_X_AAABBB.py must match
  • the name of pane in Tmux terminal window

For stop strategy use Ctrl-C Ctrl-Z and/or Telegram control function

MARGIN mode

For Linux

Extract Linux_site-packages.tar.gz

to the ~/margin-linux/resources/python/lib/python3.7/site-packages

  • Copy ms_cfg.toml and funds_rate.db to the ~/margin-linux
  • Copy executor.py to the ~/margin-linux/resources/python/lib/python3.7/site-packages

For Windows

Extract win_site-packages.zip

to the C:\Users\user\AppData\Local\Programs\margin\resources\python\lib\python3.7\site-packages

  • Copy ms_cfg.toml and funds_rate.db to the C:\Users\user\AppData\Local\Programs\margin
  • Copy executor.py to the C:\Users\user\AppData\Local\Programs\margin\resources\python\lib\python3.7\site-packages

For macOS

Extract MacOS_site-packages.zip

to the /Applications/margin-4.4.2.app/Contents/Resources/python/lib/python3.7/site-packages

  • Copy ms_cfg.toml and funds_rate.db to the /Users/current_user/.margin/
  • Copy executor.py to the /Applications/margin-4.4.2.app/Contents/Resources/python/lib/python3.7/site-packages

  • Start margin in Demo mode
  • Add currency pair BTC/USDT
  • Set the custom fee level = 0.0% in the margin terminal settings
  • Add Python strategy:

  • Copy/paste the contents of the file cli_7_BTCUSDT.py to the Code Editor window
  • Save, Run Strategy, Start

Strategy is started.

Setting trade pair. The selection of the pair is determined by the window of the terminal in which the strategy is launched. The "__ main __" section settings are ignored.

Terminal Tmux (Linux)

For STANDALONE mode you need separate terminal window for each task, separate for server and for each trading pair. If you plan to run a strategy on VPS then you need terminal program which:
  • multi windows/panes
  • continue running after end of ssh session
  • auto save running state
  • restore state after system restart

Tmux install

Install and setup all in list:

Find /service/tmux.service, edit your path and install it under systemctl.

Example for tmux configuration file .tmux.conf

set -g @plugin 'tmux-plugins/tpm'
set -g @plugin 'tmux-plugins/tmux-sensible'
set -g @plugin 'tmux-plugins/tmux-resurrect'
set -g @plugin 'tmux-plugins/tmux-continuum'
set -g @continuum-restore 'on'
set -g @continuum-save-interval '15'
set -g @continuum-boot 'on'
set -g status-right 'Status: #{continuum_status}'
set -g assume-paste-time 1
set -g base-index 0
set -g bell-action none
set -g history-file ~/.tmux_history
set -g mouse on
set -g prefix C-b
run '~/.tmux/plugins/tpm/tpm'

Attention: The name of session ('Trade') and pane ('srv', '7-BTC/USDT') must be same as in example and correspond pair which start in this pane. For example cli_1_BTCUSDT.py must be started in pane with name '1-BTC/USDT'

  • Start new session:
tmux new-session -s Trade
  • Rename pane 0: Ctrl+B + , srv Enter
  • Change dir to the ~/martin_binance
  • Create new pane: Ctrl+B c
  • Rename pane 1: Ctrl+B + , 7-BTC/USDT Enter
  • Change dir to the ~/martin_binance
  • Reboot system
  • Attach to the restored session:
tmux attach

You must see the same window as before reboot:

  • Find /service/relaunch.service, edit your path and install it under systemctl.
  • Run in the pane 0:srv server script:
./binance_srv.py
  • Run in the pane 1:7-BTC/USDT trade strategy script:
./cli_7_BTCUSDT.py

Final test, reboot system. After reboot connect to the tmux session:

tmux attach

If everything has been done correctly you should see the restored strategy.

How it's work

Setup all mentioned trade parameter at the top of cli_X_AAABBB.py

Where X is exchange number, see it in ms_cfg.toml and AAABBB is trade pair, for example BTCUSDT on Binance is cli_9_BTCUSDT.py

Place grid

Shift grid orders

Adaptive overlap price for grid

Quantity of grid orders

Fractional creation of grid

Logarithm price option

Reverse

Grid only

Place take profit order

Restart

Fee options

Telegram notification

Telegram control

Save data for external analytics

Consolidated asset valuation

Recovery after any reason crash, restart etc.

Place grid

The main parameters for the strategy are grid parameters. Specify the trade direction for the first cycle. If START_ON_BUY = True, a grid of buy orders will be placed and take profit will be for sale.

Specify the deposit size for the first cycle in the desired currency, and the number of grid orders. These are related parameters, there is a limit on the minimum size of the order, so a many orders if the deposit is insufficient will not pass the verification during initialization.

The size of the order in the grid calculated according to the law of geometric progression, while the MARTIN parameter is a coefficient of progression. The first order, the price of which is closest to the current one is the smallest in volume.

To avoid the execution of the first order "by market" its price set with a slight offset, which is determined by the parameter PRICE_SHIFT.

Shift grid orders

It happens that when place a grid, the price goes in the opposite direction. There is no point in waiting in this case, we need to move the grid after the price. For this there are SHIFT_GRID_DELAY. Configure them or leave the default values.

Adaptive overlap price for grid

The range of prices that overlaps the grid of orders affects profitability directly and must correspond to market conditions. If it is too wide combined with small number of orders, the cycle time will be too long, and most of the deposit will not be involved in turnover. With a small overlap, the entire order grid will be executed in a short time, and the algorithm will be reversed, while the profit on the cycle not fixed.

The overlap range can be fixed. Then it is defined by OVER_PRICE = xx and ADAPTIVE_TRADE_CONDITION = False.

For automatic market adjustment, ADAPTIVE_TRADE_CONDITION = True. In this case, the instant value of the Bollinger Band on 20 * 1 hour candles used to calculate the overlap range. The minimum values are limited by the exchange limit for minimal price change per one step combined with number of orders.

For fine-tuning there is KBB parameter. By default, value 2.0 is used to calculate Bollinger curves.

The over price value updated before the start of the new cycle.

Quantity of grid orders

Two parameters determine the number of orders, ORDER_Q and OVER_PRICE. For start cycle and parameter ADAPTIVE_TRADE_CONDITION = False, these are absolute and fixed values.

For Reverse cycle or parameter ADAPTIVE_TRADE_CONDITION = True they specify the density of grid orders. At the same time, the larger the range of overlap, the more orders there will be. When calculating, the exchange's restrictions on the minimum order size and price step taken into account and thus limit the maximum number of orders.

For Reverse cycle, this is especially important, since with large price fluctuations we will have to sweep a very large range, 30-50% or more. In which case an adaptive number of orders will allow to do this as efficiently as possible.

Fractional creation of grid

For successful trading, the speed of the bot response to price fluctuations is important. When testing on Bitfinex, I noticed that when placed a group of orders, the first 5 are placed quickly, and the remaining ones with a significant delay. Also, the more orders, the longer it takes to shift the grid.

Therefore, I added the parameter GRID_MAX_COUNT, which specifies the number of initial orders to be placed. Then, two options performed. First, the grid shift function triggered. Second, one of the placed grid orders executed, and after successful creation of the take profit order, the next part GRID_MAX_COUNT of hold grid orders added. If total count of active grid orders more than ORDER_Q and hold grid not exhaust each next placed one after filling one grid order, until the hold list exhausted.

Thus, we always have a given number of active orders to respond to price fluctuations and have a large range of price overlap. Then, we do not risk running into the ban from exchange for exceeding the limit.

Logarithm price option

You can increase the share of the deposit in turnover when calculating the price for grid orders using not a linear, but a logarithmic distribution. The density of orders near the current price will be higher, which increases the likelihood of their execution.

Use LINEAR_GRID_K parameter and see '/doc/Model of logarithmic grid.ods' for detail.

Reverse

It happens that all grid orders completed. Then we believe that we successfully bought the asset, and place a take profit order. Instead of one order, we place a grid, which ensures the break-even strategy. The grid parameters change dynamically, depending on the market conditions, accumulated deposit during execution of the Reverse cycle, and the specified profitability parameters.

With each successful completion of the reverse cycle, the accumulated profit volume increases, which reduces the necessary price overlap for the profitable completion of the original cycle.

In order for the algorithm not to work for itself in reverse mode, only part of the earned profit is reinvested. PROFIT_REVERSE parameter specifies the reinvestment rule. Leave it as default.

Grid only

You can buy/sell asset with grid options. To do this, set the GRID_ONLY = True In this case, the take-profit order will not be placed and Reverse cycle not started. After filling all grid orders strategy goes to STOP state. All options associated with grid calculation work as usual, except no shifting grid.

Place take profit

As the grid orders executed, the volume of the take profit order sums up their volume, price averaged and increased to override the fees and earn the specified profit.

Do not set PROFIT too large. This reduces the probability of executing a take profit order with small price fluctuations. I settled on a value of about 0.5%

For adaptive calculate profit before place order you can set PROFIT_MAX. Then its value will be set in the range from PROFIT to PROFIT_MAX. Calculation based on Bollinger Band indicator with PROFIT_K coefficient.

Restart

When take profit order executed the cycle results recorded, the deposit increased by the cycle profit, and bot restarted.

Fee options

To correctly count fees for MAKER and TAKER, you must set the FEE_MAKER and FEE_TAKER parameters.

For a third currency fee, such as BNB on Binance, or HT on Huobi, set FEE_IN_PAIR = False

For fee processing for second currency only, like as KRAKEN, use FEE_SECOND = True

Priority of parameters from larger to smaller is:

  • FEE_IN_PAIR
  • FEE_BNB_IN_PAIR
  • FEE_SECOND

Attention: the commission, which is charged in the third coin, is not taken into account in the calculation of income. Control the return on the balance of three coins.

Telegram notification

Basic information about the state of the bot, for example, about the start and results of the cycle, can be sent to Telegram bot.

Telegram control

  • status - get current status
  • stop - stop after end of cycle (if current cycle is reverse - only after back to direct cycle)
  • end - stop after end of cycle, direct and reverse, no difference

All commands are sent as a reply to message from desired strategy. Use simple text message. If all is normal, you will receive a confirmation that the system has received the command within 10 seconds.

Save data for external analytics

Analytic subsystem is not mandatory and has no impact on trading strategy.

All data collected into funds_rate.db

Cycle time, yield, funds, initial cycle parameters, all this for all pairs and exchanges where this bot launched.

In STANDALONE mode for Binance also collected all balances for all assets in trading account.

It Sqlite3 db with very simple structure, in table t_funds each row is the result of one cycle with parameters and result.

Now I'm use prometheus -> Grafana for calculate and visualisation analytic data. It's useful. You can use funds_rate_exporter.py for start. Find it into repository.

Also, you can try the grafana_config.json as example of consolidated report.

Consolidated asset valuation

If you have several trading pairs on different exchanges, then regular valuation of the asset is a very time-consuming task.

At the end of each trading cycle, deposit data recorded for each currency. Once a day, the current currency rate to USD is queried. In funds_rate_exporter.py of times per minute calculated data for unloading in prometheus. An example of a summary report implemented on Grafana located above.

To receive quotes, you need to get the free API key on CoinMarketCap. Specify the key at the top of the ms_cfg.toml and start funds_rate_exporter.py as service. For Ubuntu, you can use /service/funds_export.service

Recovery after any reason crash, restart etc.

STANDALONE mode

  • Auto for Network failure, exchange timeout etc.
  • Auto recovery after restart with full implemented Tmux install (Linux only)
  • For manual restart with save order and load last state run ./cli_X_AAABBB.py 1

margin mode

  • Network failure, timeout etc.

This is margin layer does, and it's work good. No additional action from user needed.

  • Hard restart.

Margin periodically, every two seconds, save strategy state, and after restart place strategy in suspend state. Strategy check if some order filled out during inactivity and recalculate state for normal operation. You need manual unsuspend strategy for further work.

This will work well if, at the time of interruption, the strategy was in a stable state and there were not pending actions to place/remove orders. Processing such situations requires research, and I am not sure that it is necessary.

  • Maintenance

If you need setup new version margin or Python strategy, first you need stop strategy. Use Telegram control function, described above.

For developers

The modular open architecture of the project allows you to use it both as a ready-made solution and in parts when developing your own strategies. See Review chart for reference.

Warning. Coverage of overridden margin strategy-sdk and binance.py packages is significant but not complete. Only methods and functions required for normal operation of the presented strategy are implemented. Missing functionality can be implemented on yours own or on request.

Easy way

If you want to develop 'margin style' strategy, use template from margin repository. In this way you can use it both independently (STANDALONE mode) and together with margin.

Independent way

A fully independent strategy that can only be used on the Binance SPOT Exchange.

In this case, the following modules are used: binance.py package as a layer for async communication with the Binance API.

The module binance_srv.py as a multiplexer layer, providing simultaneous async interaction for many accounts and many trading pairs through one connection from one IP address. It's powered by gRPC Remote Procedure Call framework. For Protocol Buffers serializing structured data see binance_api.proto Below code example for client, where we up connection, get client id and get info for trading pair.

import asyncio
import toml

import grpc

from google.protobuf import json_format
import binance_api_pb2
import binance_api_pb2_grpc

# For more channel options, please see https://grpc.io/grpc/core/group__grpc__arg__keys.html
CHANNEL_OPTIONS = [('grpc.lb_policy_name', 'pick_first'),
                   ('grpc.enable_retries', 0),
                   ('grpc.keepalive_timeout_ms', 10000)]
RATE_LIMITER = 10  # sec
ID_EXCHANGE = 7
FILE_CONFIG = './ms_cfg.toml'
config = toml.load(FILE_CONFIG)
EXCHANGE = config.get('exchange')
SYMBOL = 'BTCUSDT'


async def main(_symbol):
    account_name = EXCHANGE[ID_EXCHANGE]
    print(f"main.account_name: {account_name}")  # lgtm [py/clear-text-logging-sensitive-data]
    channel = grpc.aio.insecure_channel(target='localhost:50051', options=CHANNEL_OPTIONS)
    stub = binance_api_pb2_grpc.MartinStub(channel)
    client_id_msg = await stub.OpenClientConnection(binance_api_pb2.OpenClientConnectionRequest(
        account_name=account_name,
        rate_limiter=RATE_LIMITER))
    print(f"main.client_id: {client_id_msg.client_id}")
    print(f"main.srv_version: {client_id_msg.srv_version}")
    # Get symbol info based on https://th0rgal.gitbook.io/binance-py/queries/general#code-2
    _exchange_info_symbol = await stub.FetchExchangeInfoSymbol(binance_api_pb2.MarketRequest(
        client_id=client_id_msg.client_id,
        symbol=_symbol))
    exchange_info_symbol = json_format.MessageToDict(_exchange_info_symbol)
    print("\n".join(f"{k}\t{v}" for k, v in exchange_info_symbol.items()))


if __name__ == "__main__":
    loop = asyncio.get_event_loop()
    loop.run_until_complete(main(SYMBOL))
    loop.stop()
    loop.close()

Start binance_srv.py in one terminal window and this code in second.

Known issue

STANDALONE mode:

  • None

With margin.de:

  • Not work more than one Python bot at the same time with "pure virtual function" error after some time. You can use new instance for new pair
  • Sometimes skips the partial fill signal from the margin layer
  • Sometimes fill signal from the margin come with a delay, for temp fix use EXTRA_CHECK_ORDER_STATE = True
  • When you STOP strategy from margin terminal, close and then restart it, the strategy is started without approval and restore (can be duplicated) orders. If you want planed restart martin-binance - delete state.db in./margin folder before restart.

Target

  • Extended testing capabilities
  • Optimization ideas
  • Several users raise reaction from margin support
  • Resources for development
  • Get fault tolerance profitable system

Referral link

Create account on Binance and get 10% discount on all trading fee

Create account on HUOBI and get 10% cashback on all trading fee

Create account on FTX

Create account on OKEX

margin.de

Multi-exchange trade terminal. For 10% discount on margin license use coupon code Margin9ateuE

VPS

Also, you can start strategy on Hetzner cloud VPS only for 4.75 € per month

Donate

To donate and directly support the project, you can transfer funds to these addresses in Binance:

BNB, BUSD, USDT (BEP20) 0x5b52c6ba862b11318616ee6cef64388618318b92

USDT (TRC20) TP1Y43dpY7rrRyTSLaSKDZmFirqvRcpopC