• Stars
    star
    334
  • Rank 126,264 (Top 3 %)
  • Language
    Python
  • License
    The Unlicense
  • Created almost 7 years ago
  • Updated about 2 years ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

Scheduled buying of BTC, ETH, and LTC from Coinbase Pro, optimally!

optimal-buy-cbpro (formerly optimal-buy-gdax)

Build Status Maintainability Test Coverage PyPI version

Scheduled buying of BTC, ETH, and LTC from Coinbase Pro (formerly GDAX) optimally!

crypto

What is this?

This is a Python script you can use to automatically buy Bitcoin, Ethereum, Litecoin, and more using the Coinbase Pro API. By default, it buys these 3 currencies, weighted by market cap (as reported by coinmarketcap.com), using a form of dollar cost averaging according to the following logic (assuming default values):

  1. Check current balances of fiat (USD by default), BTC, ETH, and LTC
  2. If the fiat balance is above $25, buy BTC, ETH, and LTC weighted by market cap, as follows:
    • If there's enough fiat available, place 5 discounted limit orders at the current price minus 0.5% up to 4.5%, each order with 1/5th of the remaining amount to buy for each coin (see "Details on the orders placed", below)
    • If there isn't enough USD available, place 1 buy order at 0.5% off the current price (see "Order Minimums")
  3. If the fiat account balance is below $25 (or whatever you specify), withdraw coins to desired addresses

In effect, this script mimmicks the behaviour of a market cap weighted index fund, but without the fees. It also only supports the coins that trade on Coinbase Pro (because that's the only exchange that has an API for ACH deposits AFAIK).

You can also use the same script to schedule deposits from your bank account periodically, such as when you're paid. The parameters may be configured to suit your preferences, such as which coins to buy, external balances, discount values, number of steps, etc.

Orders, deposits, and withdrawals are tracked in a SQLite DB, and the withdrawn balances are added to the balances on Coinbase Pro to make sure the weights are maintained over time. The SQLite DB can be swapped out for any DB that SQLAlchemy supports.

A note on the default parameters: it's likely you'll want to change --starting-discount, --discount-step, or --order-count. The more spread out the orders are (i.e., difference between the current price and the lowest priced order), the longer they will take to fill (if they fill), and the closer the orders are, the more likely you are to miss out on bigger price drops. You should consider your appetite for risk and how much you want to optimize for catching those dips vs. not missing out on gains. There is no magic here. My personal advice is to stick somewhat close to the defaults, and try to continuously deposit a little more fiat every week to spread the risk but also catch some dips.

Ideally, this script would help to make sure that when we dip—

dip

we buy.

USE AT OWN RISK

Duh. Not my fault if you lose everything.

Unless you place absolute trust in me, some guy from the Internet, I suggest you clone the repo and build your own container to protect yourself from any type of funny business.

How do I use it?

The package itself can be used as a Docker container, or by installing the pip package with pip install optimal-buy-cbpro. Using the Docker container is recommended to avoid Python environment issues. Instructions for running with Docker and systemd are as follows:

  1. Get yourself a hardware wallet, such as a Ledger or TREZOR.

  2. Set up a Coinbase Pro account, and link your bank account

  3. Create a Coinbase Pro API key with view, trade, manage, transfer, and bypass-2fa permissions

  4. Determine the payment_method_id value by using the Coinbase Pro API (you can use your browser's developer toolbar, here's a quick video showing how)

  5. Get a machine somewhere (GCE, EC2, Digital Ocean) with Docker and systemd

  6. Copy systemd files over:

    $ sudo cp systemd/optimal-buy-cbpro-*.{service,timer} /etc/systemd/system
    
  7. Edit /etc/systemd/system/optimal-buy-cbpro-buy.service, /etc/systemd/system/optimal-buy-cbpro-buy.timer, /etc/systemd/system/optimal-buy-cbpro-deposit.service, and /etc/systemd/system/optimal-buy-cbpro-deposit.timer to your liking. Make sure you:

    • Change the BTC, ETH, and LTC withdrawal addresses to deposit the coins into your wallet (use a Ledger or TREZOR)
    • Pop in the correct API keys
    • Check the deposit amount (start with something small, like $150, to make sure it actually works first)
    • Check the timer dates (it would be sensible to change the hh:mm so your script doesn't run the same time as everyone else's), make sure the deposit timer fires according to your deposit schedule (keeping in mind that ACH takes 2-5 business days to clear, typically)
    • Consider specifying your external balances in order to accurately calculate the weights and amounts to purchase
  8. Enable the systemd units:

    $ sudo systemctl enable optimal-buy-cbpro-buy.service
    $ sudo systemctl enable optimal-buy-cbpro-buy.timer
    $ sudo systemctl enable optimal-buy-cbpro-deposit.service
    $ sudo systemctl enable optimal-buy-cbpro-deposit.timer
    
  9. Start the systemd timers:

    $ sudo systemctl start optimal-buy-cbpro-buy.timer
    $ sudo systemctl start optimal-buy-cbpro-deposit.timer
    
  10. Enjoy!

Configuration

usage: optimal-buy-cbpro [-h] --mode MODE [--amount AMOUNT] --key KEY
                        --b64secret B64SECRET --passphrase PASSPHRASE
                        [--api-url API_URL]
                        [--payment-method-id PAYMENT_METHOD_ID]
                        [--starting-discount STARTING_DISCOUNT]
                        [--discount-step DISCOUNT_STEP]
                        [--order-count ORDER_COUNT]
                        [--fiat-currency FIAT_CURRENCY]
                        [--withdrawal-amount WITHDRAWAL_AMOUNT]
                        [--db-engine DB_ENGINE] [--max-retries MAX_RETRIES]
                        [--coins COINS] [--base-fee BASE_FEE]

Buy coins!

optional arguments:
  -h, --help            show this help message and exit
  --mode MODE           mode (deposit or buy)
  --amount AMOUNT       amount to deposit
  --key KEY             API key
  --b64secret B64SECRET
                        API secret
  --passphrase PASSPHRASE
                        API passphrase
  --api-url API_URL     API URL (default: https://api.pro.coinbase.com)
  --payment-method-id PAYMENT_METHOD_ID
                        Payment method ID for fiat deposits
  --starting-discount STARTING_DISCOUNT
                        starting discount (default: 0.005)
  --discount-step DISCOUNT_STEP
                        discount step between orders (default: 0.01)
  --order-count ORDER_COUNT
                        number of orders (default: 5)
  --fiat-currency FIAT_CURRENCY
                        Fiat currency (default: USD)
  --withdrawal-amount WITHDRAWAL_AMOUNT
                        withdraw when fiat balance drops below this amount
                        (default: 25)
  --db-engine DB_ENGINE
                        SQLAlchemy DB engine (default:
                        sqlite:///cbpro_history.db)
  --max-retries MAX_RETRIES
                        Maximum number of times to retry if there are any
                        failures, such as API issues (default: 3)
  --coins COINS         Coins to trade, minimum trade size, withdrawal
                        addresses and external balances. Accepts a JSON
                        string.
  --base-fee BASE_FEE   Default base fee to subtract from overall balance.

Default coins are as follows:
    {
      "BTC":{
        "name":"Bitcoin",
        "withdrawal_address":null,
        "external_balance":0
      },
      "ETH":{
        "name":"Ethereum",
        "withdrawal_address":null,
        "external_balance":0
      },
      "LTC":{
        "name":"Litecoin",
        "withdrawal_address":null,
        "external_balance":0
      }
    }

Details on the orders placed

By default, there are 5 orders placed (for each currency) in steps of 1%, starting at a 0.5% discount from the current price. To illustrate, if the current price was $100 (per LTC, let's say), and you had$100 to buy, the orders would look like this:

Order Size Price
1 0.2010 LTC $99.5
2 0.2030 LTC $98.5
3 0.2051 LTC $97.5
4 0.2072 LTC $96.5
5 0.2094 LTC $95.5

Furthermore, the amount of each currency to buy will be based on the current market cap weighting of each coin. For example, at the time of writing the weights are:

Coin Market Cap (USD) Weight
BTC $195,824,365,435 0.791
ETH $46,080,472,372 0.186
LTC $5,592,776,540 0.023

So if your USD account had $1000, the amount purchased of each would become:

Coin Weight Amount Purchased
BTC 0.791 $791
ETH 0.186 $186
LTC 0.023 $23

Caveats/limitations

  • If you try to trade manually or using some other bot at the same time, you're probably going to have a bad time
  • You might have a few dollars (<$25, you can change this with --withdrawal-amount) sitting in your account at all times, even when all orders have been filled because it's not always possible to fill all orders and there may be small rounding errors (on the order of cents)
  • It makes a best effort with minimal complexity to use all of your fiat, but it may not be possible to fill all orders right away
  • It may take a few days for the market to drop enough for the buys to fill
  • If the market experiences a significant bull run, your orders won't be filled, but it will reset every 24h (using the default buy timer)

More Repositories

1

conky

Light-weight system monitor for X, Wayland (sort of), and other things, too
C++
7,128
star
2

thetagang

ThetaGang is an IBKR bot for collecting money
Python
1,930
star
3

cracking-the-coding-interview-rust

Cracking the Coding Interview problem solutions in Rust
Rust
394
star
4

facebook-hive-udfs

Facebook's Hive UDFs
Java
270
star
5

dryoc

Don't Roll Your Own Crypto: pure-Rust, hard to misuse cryptography library
Rust
266
star
6

tweet-delete

Self-destructing Tweets so you too can be cool 😎
Python
94
star
7

code-like-a-pro-in-rust-book

Source code for Code Like a Pro in Rust
HTML
93
star
8

idiomatic-rust-book

Source code for Idiomatic Rust: Code like a Rustacean
Rust
78
star
9

rust-react-typescript-demo

Demo for Rust, React, Typescript, Docker, Terraform and Kubernetes
CSS
68
star
10

nginx-echo-headers

Have nginx return request headers to the client
Dockerfile
61
star
11

vault-dcos

Vault on DCOS
Shell
45
star
12

hodlermanifesto

The HODLer Manifesto
SCSS
38
star
13

labhub

GitHub bot for using GitLab CI in OSS projects
Rust
32
star
14

seed-otp

One-time pad tool for Bitcoin seed mnemonic cold storage
Python
21
star
15

metrics-cassandra

Dropwizard Metrics Cassandra reporter
Java
17
star
16

mother-of-dragons

🐲 DragonMint/Innosilicon miner management tool 🐉
Python
17
star
17

marathon-lb-autoscale

Autoscale your apps on Marathon
Ruby
14
star
18

kafka-on-marathon

Scripts for running Apache Kafka on Mesosphere's Marathon
HTML
14
star
19

citrine

Elixir library for running cron-based scheduled jobs on your Erlang cluster
Elixir
14
star
20

genserver

Elixir inspired async actor library for Rust
Rust
13
star
21

omploader

Enterprise quality file hosting web application.
Ruby
12
star
22

cgminer-rest

RESTful HTTP API wrapper for cgminer
Rust
9
star
23

hdfs

HA HDFS on Apache Mesos ~~~ aka 'Super Available' HDFS
Java
9
star
24

mlb-nginx-http2

Demo of marathon-lb with nginx and HTTP/2
Nginx
9
star
25

protect-yourself

A guide on how to protect your digital assets
HTML
9
star
26

dragon-rest

🐲 Python wrapper for DragonMint/Innosilicon REST API 🐉
Python
8
star
27

rust-action

All-in-one GitHub action for Rust projects
Rust
8
star
28

startup-ideas

Some startup ideas I'm considering
7
star
29

supertrees

Experimental Erlang/OTP-inspired supervision trees for Rust
Rust
6
star
30

resume

My Resume
TeX
6
star
31

brndn-io

It is my website
CSS
6
star
32

rust-action-cargo-binstall

GitHub action for installing Cargo packages
TypeScript
4
star
33

elasticsearch-router

An nginx based router for Elasticsearch on DCOS
Nginx
3
star
34

rust-action-rustup

GitHub action for installing rustup
TypeScript
3
star
35

grafana-dcos

Grafana on DCOS backed by Elasticsearch
3
star
36

cassandra-on-marathon

Scripts for running Apache Cassandra on Mesosphere's Marathon
Ruby
3
star
37

nginx-echo-sleep

Nginx
2
star
38

box-with-nonce

Rust
2
star
39

doge-streamer

Stream Doge
C++
2
star
40

nginx-redirect-to-https

Redirect all HTTP traffic to HTTPS (for use with Marathon & Marathon-lb)
Nginx
1
star
41

doge-stream-helper

Helper app for the Doge stream
HTML
1
star
42

dnstest

DNS client test tools
C
1
star
43

gitlab-gce-autoscaler

Simple GCE autoscaler for GitLab CI pipelines
Python
1
star
44

gwyh

Library for building gossip-based services in Rust
Rust
1
star