mirror of
https://github.com/d0zingcat/gocryptotrader.git
synced 2026-05-29 15:10:37 +00:00
* Adds some constants for fee types Adds some fee calculation in an attempt to be generic Adds fee stuff to Bittrex Adds fee stuff to bitstamp * Fixes bitstamp fee calculation * Tests Tests all scenarios for GetFeeByType * Adds method to wrapper Adds err to response Checks for err * Adds support for Bittrex fees * Adds maker/taker dynamic to fees Updates tests Adds bitmex fee support Removes unused switch case scenarios to not waste space * Adds bithumb support for fee calculation * Adds Bitfinex fee support Adds list of currencies as const strings Sets up bitflyer * Fixes arguments * Greatly expands symbols Adds Binance fee calculation support Cleans up previous exchanges * Fixes errors for fee calculations * Adds ANX fee support * Adds btcc fee support Adds alphapoint fee wrapper support Renames method to match "enum" Uses symbols in tests, not inline strings * Adds support for BTCMarkets fee calculation Adds new method to retrieve fee amount from BTCMarkets Adds new fee type struct: FeeBuilder Updates ANX and BTCMarkets to use new FeeBuilder type struct Standardises the tests to run when it comes to fee calculation * Migrates all existing exchange fee to use new feebuilder type struct Uses standard testing model * Fixes unit tests * Updates maker taker fees in test config * Removes parallel from fee testing * Removes more parallel from tests * Adds coinbasepro fee support * Adds Coinut fee support * Adds Exmo fee support Adds maker fee support to coinut Introduces a type for fees and bank transfers to prevent random strings being used * Adds partial bitflyer support Moves bitflyer to feeBuilder struct * Adds gateio fee support * Adds Gemini fee support * Adds hitbtc fee support * Adds huobi fee support * Adds HuobiHadax fee support * Adds itbit fee support * Adds partial kraken fee support with trading fees * Finishes basic Kraken fee support * Adds basic LakeBTC fee support * Adds basic liqui fee support * Adds localbitcoins fee support....... * Adds basic okcoin fee support * Adds simple OKEX fee support Adds many new currency symbols Fixes liqui's fees * Adds poloniex fee support * Adds fee support for Yobit * Adds WEX fee support * Adds ZB fee support * Removes bad reference * Improves accuracy of variable name * trading fee method names are now consistent (cherry picked from commit 21c82e8b90cae590cfd73d365d7be39e1a00e973) * Fixes rebasing issues * Fixes issues from rebase Removes "IsTaker" as IsMaker bool can imply taker Updates tests to actually work. * Adds a zero to the test * Fixes bitfinex api endpoints and fixes fee calculations * Updates btcmarkets trading fee calculation * Verifies tests with apis for all exchanges except coinbasepro, itbit and bitflyer Removes taker fee test as taker is default * Removes redundant all exchange wrapper error checks due to the error checks being redundant * Addresses review comments: - Renames variables - Changes how functions return data - Fixes typo
414 lines
12 KiB
Go
414 lines
12 KiB
Go
package coinut
|
|
|
|
import (
|
|
"bytes"
|
|
"errors"
|
|
"fmt"
|
|
"log"
|
|
"time"
|
|
|
|
"github.com/gorilla/websocket"
|
|
"github.com/thrasher-/gocryptotrader/common"
|
|
"github.com/thrasher-/gocryptotrader/config"
|
|
"github.com/thrasher-/gocryptotrader/currency"
|
|
"github.com/thrasher-/gocryptotrader/currency/symbol"
|
|
"github.com/thrasher-/gocryptotrader/exchanges"
|
|
"github.com/thrasher-/gocryptotrader/exchanges/request"
|
|
"github.com/thrasher-/gocryptotrader/exchanges/ticker"
|
|
)
|
|
|
|
const (
|
|
coinutAPIURL = "https://api.coinut.com"
|
|
coinutAPIVersion = "1"
|
|
coinutInstruments = "inst_list"
|
|
coinutTicker = "inst_tick"
|
|
coinutOrderbook = "inst_order_book"
|
|
coinutTrades = "inst_trade"
|
|
coinutBalance = "user_balance"
|
|
coinutOrder = "new_order"
|
|
coinutOrders = "new_orders"
|
|
coinutOrdersOpen = "user_open_orders"
|
|
coinutOrderCancel = "cancel_order"
|
|
coinutOrdersCancel = "cancel_orders"
|
|
coinutTradeHistory = "trade_history"
|
|
coinutIndexTicker = "index_tick"
|
|
coinutOptionChain = "option_chain"
|
|
coinutPositionHistory = "position_history"
|
|
coinutPositionOpen = "user_open_positions"
|
|
|
|
coinutAuthRate = 0
|
|
coinutUnauthRate = 0
|
|
)
|
|
|
|
// COINUT is the overarching type across the coinut package
|
|
type COINUT struct {
|
|
exchange.Base
|
|
WebsocketConn *websocket.Conn
|
|
InstrumentMap map[string]int
|
|
}
|
|
|
|
// SetDefaults sets current default values
|
|
func (c *COINUT) SetDefaults() {
|
|
c.Name = "COINUT"
|
|
c.Enabled = false
|
|
c.Verbose = false
|
|
c.TakerFee = 0.1 //spot
|
|
c.MakerFee = 0
|
|
c.Verbose = false
|
|
c.RESTPollingDelay = 10
|
|
c.RequestCurrencyPairFormat.Delimiter = ""
|
|
c.RequestCurrencyPairFormat.Uppercase = true
|
|
c.ConfigCurrencyPairFormat.Delimiter = ""
|
|
c.ConfigCurrencyPairFormat.Uppercase = true
|
|
c.AssetTypes = []string{ticker.Spot}
|
|
c.SupportsAutoPairUpdating = true
|
|
c.SupportsRESTTickerBatching = false
|
|
c.Requester = request.New(c.Name,
|
|
request.NewRateLimit(time.Second, coinutAuthRate),
|
|
request.NewRateLimit(time.Second, coinutUnauthRate),
|
|
common.NewHTTPClientWithTimeout(exchange.DefaultHTTPTimeout))
|
|
c.APIUrlDefault = coinutAPIURL
|
|
c.APIUrl = c.APIUrlDefault
|
|
c.WebsocketInit()
|
|
}
|
|
|
|
// Setup sets the current exchange configuration
|
|
func (c *COINUT) Setup(exch config.ExchangeConfig) {
|
|
if !exch.Enabled {
|
|
c.SetEnabled(false)
|
|
} else {
|
|
c.Enabled = true
|
|
c.AuthenticatedAPISupport = exch.AuthenticatedAPISupport
|
|
c.SetAPIKeys(exch.APIKey, exch.APISecret, exch.ClientID, true)
|
|
c.SetHTTPClientTimeout(exch.HTTPTimeout)
|
|
c.SetHTTPClientUserAgent(exch.HTTPUserAgent)
|
|
c.RESTPollingDelay = exch.RESTPollingDelay
|
|
c.Verbose = exch.Verbose
|
|
c.Websocket.SetEnabled(exch.Websocket)
|
|
c.BaseCurrencies = common.SplitStrings(exch.BaseCurrencies, ",")
|
|
c.AvailablePairs = common.SplitStrings(exch.AvailablePairs, ",")
|
|
c.EnabledPairs = common.SplitStrings(exch.EnabledPairs, ",")
|
|
err := c.SetCurrencyPairFormat()
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
err = c.SetAssetTypes()
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
err = c.SetAutoPairDefaults()
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
err = c.SetAPIURL(exch)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
err = c.SetClientProxyAddress(exch.ProxyAddress)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
err = c.WebsocketSetup(c.WsConnect,
|
|
exch.Name,
|
|
exch.Websocket,
|
|
coinutWebsocketURL,
|
|
exch.WebsocketURL)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
}
|
|
}
|
|
|
|
// GetInstruments returns instruments
|
|
func (c *COINUT) GetInstruments() (Instruments, error) {
|
|
var result Instruments
|
|
params := make(map[string]interface{})
|
|
params["sec_type"] = "SPOT"
|
|
|
|
return result, c.SendHTTPRequest(coinutInstruments, params, false, &result)
|
|
}
|
|
|
|
// GetInstrumentTicker returns a ticker for a specific instrument
|
|
func (c *COINUT) GetInstrumentTicker(instrumentID int) (Ticker, error) {
|
|
var result Ticker
|
|
params := make(map[string]interface{})
|
|
params["inst_id"] = instrumentID
|
|
|
|
return result, c.SendHTTPRequest(coinutTicker, params, false, &result)
|
|
}
|
|
|
|
// GetInstrumentOrderbook returns the orderbooks for a specific instrument
|
|
func (c *COINUT) GetInstrumentOrderbook(instrumentID, limit int) (Orderbook, error) {
|
|
var result Orderbook
|
|
params := make(map[string]interface{})
|
|
params["inst_id"] = instrumentID
|
|
if limit > 0 {
|
|
params["top_n"] = limit
|
|
}
|
|
|
|
return result, c.SendHTTPRequest(coinutOrderbook, params, false, &result)
|
|
}
|
|
|
|
// GetTrades returns trade information
|
|
func (c *COINUT) GetTrades(instrumentID int) (Trades, error) {
|
|
var result Trades
|
|
params := make(map[string]interface{})
|
|
params["inst_id"] = instrumentID
|
|
|
|
return result, c.SendHTTPRequest(coinutTrades, params, false, &result)
|
|
}
|
|
|
|
// GetUserBalance returns the full user balance
|
|
func (c *COINUT) GetUserBalance() (UserBalance, error) {
|
|
result := UserBalance{}
|
|
|
|
return result, c.SendHTTPRequest(coinutBalance, nil, true, &result)
|
|
}
|
|
|
|
// NewOrder places a new order on the exchange
|
|
func (c *COINUT) NewOrder(instrumentID int, quantity, price float64, buy bool, orderID uint32) (interface{}, error) {
|
|
var result interface{}
|
|
params := make(map[string]interface{})
|
|
params["inst_id"] = instrumentID
|
|
params["price"] = price
|
|
params["qty"] = quantity
|
|
params["side"] = "BUY"
|
|
if !buy {
|
|
params["side"] = "SELL"
|
|
}
|
|
params["client_ord_id"] = orderID
|
|
|
|
return result, c.SendHTTPRequest(coinutOrder, params, true, &result)
|
|
}
|
|
|
|
// NewOrders places multiple orders on the exchange
|
|
func (c *COINUT) NewOrders(orders []Order) ([]OrdersBase, error) {
|
|
var result OrdersResponse
|
|
params := make(map[string]interface{})
|
|
params["orders"] = orders
|
|
|
|
return result.Data, c.SendHTTPRequest(coinutOrders, params, true, &result.Data)
|
|
}
|
|
|
|
// GetOpenOrders returns a list of open order and relevant information
|
|
func (c *COINUT) GetOpenOrders(instrumentID int) ([]OrdersResponse, error) {
|
|
var result []OrdersResponse
|
|
params := make(map[string]interface{})
|
|
params["inst_id"] = instrumentID
|
|
|
|
return result, c.SendHTTPRequest(coinutOrdersOpen, params, true, &result)
|
|
}
|
|
|
|
// CancelOrder cancels a specific order and returns if it was actioned
|
|
func (c *COINUT) CancelOrder(instrumentID, orderID int) (bool, error) {
|
|
var result GenericResponse
|
|
params := make(map[string]interface{})
|
|
params["inst_id"] = instrumentID
|
|
params["order_id"] = orderID
|
|
|
|
err := c.SendHTTPRequest(coinutOrdersCancel, params, true, &result)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
return true, nil
|
|
}
|
|
|
|
// CancelOrders cancels multiple orders
|
|
func (c *COINUT) CancelOrders(orders []CancelOrders) (CancelOrdersResponse, error) {
|
|
var result CancelOrdersResponse
|
|
params := make(map[string]interface{})
|
|
params["entries"] = orders
|
|
|
|
return result, c.SendHTTPRequest(coinutOrdersCancel, params, true, &result)
|
|
}
|
|
|
|
// GetTradeHistory returns trade history for a specific instrument.
|
|
func (c *COINUT) GetTradeHistory(instrumentID, start, limit int) (TradeHistory, error) {
|
|
var result TradeHistory
|
|
params := make(map[string]interface{})
|
|
params["inst_id"] = instrumentID
|
|
if start >= 0 && start <= 100 {
|
|
params["start"] = start
|
|
}
|
|
if limit >= 0 && start <= 100 {
|
|
params["limit"] = limit
|
|
}
|
|
|
|
return result, c.SendHTTPRequest(coinutTradeHistory, params, true, &result)
|
|
}
|
|
|
|
// GetIndexTicker returns the index ticker for an asset
|
|
func (c *COINUT) GetIndexTicker(asset string) (IndexTicker, error) {
|
|
var result IndexTicker
|
|
params := make(map[string]interface{})
|
|
params["asset"] = asset
|
|
|
|
return result, c.SendHTTPRequest(coinutIndexTicker, params, false, &result)
|
|
}
|
|
|
|
// GetDerivativeInstruments returns a list of derivative instruments
|
|
func (c *COINUT) GetDerivativeInstruments(secType string) (interface{}, error) {
|
|
var result interface{} //to-do
|
|
params := make(map[string]interface{})
|
|
params["sec_type"] = secType
|
|
|
|
return result, c.SendHTTPRequest(coinutInstruments, params, false, &result)
|
|
}
|
|
|
|
// GetOptionChain returns option chain
|
|
func (c *COINUT) GetOptionChain(asset, secType string, expiry int64) (OptionChainResponse, error) {
|
|
var result OptionChainResponse
|
|
params := make(map[string]interface{})
|
|
params["asset"] = asset
|
|
params["sec_type"] = secType
|
|
|
|
return result, c.SendHTTPRequest(coinutOptionChain, params, false, &result)
|
|
}
|
|
|
|
// GetPositionHistory returns position history
|
|
func (c *COINUT) GetPositionHistory(secType string, start, limit int) (PositionHistory, error) {
|
|
var result PositionHistory
|
|
params := make(map[string]interface{})
|
|
params["sec_type"] = secType
|
|
if start >= 0 {
|
|
params["start"] = start
|
|
}
|
|
if limit >= 0 {
|
|
params["limit"] = limit
|
|
}
|
|
|
|
return result, c.SendHTTPRequest(coinutPositionHistory, params, true, &result)
|
|
}
|
|
|
|
// GetOpenPositions returns all your current opened positions
|
|
func (c *COINUT) GetOpenPositions(instrumentID int) ([]OpenPosition, error) {
|
|
type Response struct {
|
|
Positions []OpenPosition `json:"positions"`
|
|
}
|
|
var result Response
|
|
params := make(map[string]interface{})
|
|
params["inst_id"] = instrumentID
|
|
|
|
return result.Positions,
|
|
c.SendHTTPRequest(coinutPositionOpen, params, true, &result)
|
|
}
|
|
|
|
//to-do: user position update via websocket
|
|
|
|
// SendHTTPRequest sends either an authenticated or unauthenticated HTTP request
|
|
func (c *COINUT) SendHTTPRequest(apiRequest string, params map[string]interface{}, authenticated bool, result interface{}) (err error) {
|
|
if !c.AuthenticatedAPISupport && authenticated {
|
|
return fmt.Errorf(exchange.WarningAuthenticatedRequestWithoutCredentialsSet, c.Name)
|
|
}
|
|
|
|
if c.Nonce.Get() == 0 {
|
|
c.Nonce.Set(time.Now().Unix())
|
|
} else {
|
|
c.Nonce.Inc()
|
|
}
|
|
|
|
if params == nil {
|
|
params = map[string]interface{}{}
|
|
}
|
|
params["nonce"] = c.Nonce.Get()
|
|
params["request"] = apiRequest
|
|
|
|
payload, err := common.JSONEncode(params)
|
|
if err != nil {
|
|
return errors.New("SenddHTTPRequest: Unable to JSON request")
|
|
}
|
|
|
|
if c.Verbose {
|
|
log.Printf("Request JSON: %s\n", payload)
|
|
}
|
|
|
|
headers := make(map[string]string)
|
|
if authenticated {
|
|
headers["X-USER"] = c.ClientID
|
|
hmac := common.GetHMAC(common.HashSHA256, []byte(payload), []byte(c.APIKey))
|
|
headers["X-SIGNATURE"] = common.HexEncodeToString(hmac)
|
|
}
|
|
headers["Content-Type"] = "application/json"
|
|
|
|
return c.SendPayload("POST", c.APIUrl, headers, bytes.NewBuffer(payload), result, authenticated, c.Verbose)
|
|
}
|
|
|
|
// GetFee returns an estimate of fee based on type of transaction
|
|
func (c *COINUT) GetFee(feeBuilder exchange.FeeBuilder) (float64, error) {
|
|
var fee float64
|
|
switch feeBuilder.FeeType {
|
|
case exchange.CryptocurrencyTradeFee:
|
|
fee = c.calculateTradingFee(feeBuilder.FirstCurrency, feeBuilder.SecondCurrency, feeBuilder.PurchasePrice, feeBuilder.Amount, feeBuilder.IsMaker)
|
|
case exchange.InternationalBankWithdrawalFee:
|
|
fee = getInternationalBankWithdrawalFee(feeBuilder.CurrencyItem, feeBuilder.Amount)
|
|
case exchange.InternationalBankDepositFee:
|
|
fee = getInternationalBankDepositFee(feeBuilder.CurrencyItem, feeBuilder.Amount)
|
|
}
|
|
|
|
if fee < 0 {
|
|
fee = 0
|
|
}
|
|
|
|
return fee, nil
|
|
}
|
|
|
|
func (c *COINUT) calculateTradingFee(firstCurrency, secondCurrency string, purchasePrice, amount float64, isMaker bool) float64 {
|
|
var fee float64
|
|
if isMaker {
|
|
fee = 0
|
|
} else if currency.IsCryptocurrency(firstCurrency) && !currency.IsCryptocurrency(secondCurrency) ||
|
|
!currency.IsCryptocurrency(firstCurrency) && currency.IsCryptocurrency(secondCurrency) {
|
|
fee = 0.002
|
|
} else {
|
|
fee = 0.001
|
|
}
|
|
|
|
return fee * amount * purchasePrice
|
|
}
|
|
|
|
func getInternationalBankWithdrawalFee(currency string, amount float64) float64 {
|
|
var fee float64
|
|
|
|
if currency == symbol.USD {
|
|
if amount*0.001 < 10 {
|
|
fee = 10
|
|
} else {
|
|
fee = amount * 0.001
|
|
}
|
|
} else if currency == symbol.CAD {
|
|
if amount*0.005 < 10 {
|
|
fee = 2
|
|
} else {
|
|
fee = amount * 0.005
|
|
}
|
|
} else if currency == symbol.SGD {
|
|
if amount*0.001 < 10 {
|
|
fee = 10
|
|
} else {
|
|
fee = amount * 0.001
|
|
}
|
|
}
|
|
|
|
return fee
|
|
}
|
|
|
|
func getInternationalBankDepositFee(currency string, amount float64) float64 {
|
|
var fee float64
|
|
|
|
if currency == symbol.USD {
|
|
if amount*0.001 < 10 {
|
|
fee = 10
|
|
} else {
|
|
fee = amount * 0.001
|
|
}
|
|
} else if currency == symbol.CAD {
|
|
if amount*0.005 < 10 {
|
|
fee = 2
|
|
} else {
|
|
fee = amount * 0.005
|
|
}
|
|
}
|
|
|
|
return fee
|
|
}
|