mirror of
https://github.com/d0zingcat/gocryptotrader.git
synced 2026-05-18 07:26:50 +00:00
* implements futures functions and GRPC functions on new branch * lint and test fixes * Fix uneven split pnl. Adds collateral weight test. docs. New clear func * Test protection if someone has zero collateral * Uses string instead of double for accuracy * Fixes old code panic * context, match, docs * Addresses Shazniterinos, var names, expanded tests * Returns subaccount name, provides USD values when offlinecalc * Fixes oopsie * Fixes cool bug which allowed made up subaccount results * Subaccount override on FTX, subaccount results for collateral * Strenghten collateral account info checks. Improve FTX test * English is my first language * Fixes oopsies * Adds some conceptual futures order details to track PNL * Initial design of future order processing in the backtester * Introduces futures concept for collateral and spot/futures config diffs * Fixes most tests * Simple designs for collateral funding pair concept * Expands interface use so much it hurts * Implements more collateral interfaces * Adds liquidation, adds strategy, struggles with Binance * Attempts at getting FTX to work * Adds calculatePNL as a wrapper function and adds an `IsFutures` asset check * Successfully loads backtester with collateral currency * Fails to really get much going for supporting futures * Merges master changes * Fleshes out how FTX processes collateral * Further FTX collateral workings * hooks up more ftx collateral and pnl calculations * more funcs to flesh out handling * Adds more links, just can't fit the pieces together :( * Greatly expands futures order processing * Fleshes out position tracker to also handle asset and exchange +testing * RM linkedOrderID. rn positioncontroller, unexport * Successfully tracks futures order positions * Fails to calculate PNL * Calculates pnl from orders accurately with exception to flipping orders * Calculates PNL from orders * Adds another controller layer to make it ez from orderstore * Backtester now compiles. Adds test coverage * labels things add scaling collateral test * Calculates pnl in line with fees * Mostly accurate PNL, with exception to appending with diff prices * Adds locks, adds rpc function * grpc implementations * Gracefully handles rpc function * beautiful tests! * rejiggles tests to polish * Finishes FTX testing, adds comments * Exposes collateral calculations to rpc * Adds commands and testing for rpcserver.go functions * Increase testing and fix up backtester code * Returns cool changes to original branch * end of day fixes * Fixing some tests * Fixing tests 🎉 * Fixes all the tests * Splits the backtester setup and running into different files * Merge, minor fixes * Messing with some strategy updates * Failed understanding at collateral usage * Begins the creation of cash and carry strategy * Adds underlying pair, adds filldependentevent for futures * Completes fill prerequsite event implementation. Can't short though * Some bug fixes * investigating funds * CAN NOW CREATE A SHORT ORDER * Minor change in short size * Fixes for unrealised PNL & collateral rendering * Fixes lint and tests * Adds some verbosity * Updates to pnl calc * Tracks pnl for short orders, minor update to strategy * Close and open event based on conditions * Adds pnl data for currency statistics * Working through PNL calculation automatically. Now panics * Adds tracking, is blocked from design * Work to flesh out closing a position * vain attempts at tracking zeroing out bugs * woww, super fun new subloggers 🎉 * Begins attempt at automatically handling contracts and collateral based on direction * Merge master + fixes * Investigating issues with pnl and holdings * Minor pnl fixes * Fixes future position sizing, needs contract sizing * Can render pnl results, focussing on funding statistics * tracking candles for futures, but why not btc * Improves funding statistics * Colours and stats * Fixes collateral and snapshot bugs * Completes test * Fixes totals bug * Fix double buy, expand stats, fixes usd totals, introduce interface * Begins report formatting and calculations * Appends pnl to receiving curr. Fixes map[time]. accurate USD * Improves report output rendering * PNL stats in report. New tests for futures * Fixes existing tests before adding new coverage * Test coverage * Completes portfolio coverage * Increase coverage exchange, portfolio. fix size bug. NEW CHART * WHAT IS GOING ON WITH PNL * Fixes PNL calculation. Adds ability to skip om futures tracking * minor commit before merge * Adds basic liquidation to backtester * Changes liquidation to order based * Liquidationnnnnn * Further fleshes out liquidations * Completes liquidations in a honorable manner. Adds AppendReasonf * Beginnings of spot futures gap chart. Needs to link currencies to render difference * Removes fake liquidation. Adds cool new chart * Fixes somet tests,allows for zero fee value v nil distinction,New tests * Some annoying test fixes that took too long * portfolio coverage * holding coverage, privatisation funding * Testwork * boring tests * engine coverage * More backtesting coverage * Funding, strategy, report test coverage * Completes coverage of report package * Documentation, fixes some assumptions on asset errors * Changes before master merge * Lint and Tests * defaults to non-coloured rendering * Chart rendering * Fixes surprise non-local-lints * Niterinos to the extremeos * Fixes merge problems * The linter splintered across the glinting plinths * Many nits addressed. Now sells spot position on final candle * Adds forgotten coverage * Adds ability to size futures contracts to match spot positions. * fixes order sell sizing * Adds tests to sizing. Fixes charting issue * clint splintered the linters with flint * Improves stats, stat rendering * minifix * Fixes tests and fee bug * Merge fixeroos * Microfixes * Updates orderPNL on first Correctly utilises fees. Adds committed funds * New base funcs. New order summary * Fun test updates * Fix logo colouring * Fixes niteroonies * Fix report * BAD COMMIT * Fixes funding issues.Updates default fee rates.Combines cashcarry case * doc regen * Now returns err * Fixes sizing bug issue introduced in PR * Fixes fun fee/total US value bug * Fix chart bug. Show log charts with disclaimer * sellside fee * fixes fee and slippage view * Fixed slippage price issue * Fixes calculation and removes rendering * Fixes stats and some rendering * Merge fix * Fixes merge issues * go mod tidy, lint updates * New linter attempt * Version bump in appveyor and makefile * Regex filename, config fixes, template h2 fixes * Removes bad stats. * neatens config builder. Moves filename generator * Fixes issue where linter wants to fix my spelling * Fixes pointers and starts
1074 lines
28 KiB
Go
1074 lines
28 KiB
Go
package order
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/shopspring/decimal"
|
|
"github.com/thrasher-corp/gocryptotrader/common"
|
|
"github.com/thrasher-corp/gocryptotrader/currency"
|
|
"github.com/thrasher-corp/gocryptotrader/exchanges/asset"
|
|
)
|
|
|
|
const testExchange = "test"
|
|
|
|
// FakePNL implements PNL interface
|
|
type FakePNL struct {
|
|
err error
|
|
result *PNLResult
|
|
}
|
|
|
|
// CalculatePNL overrides default pnl calculations
|
|
func (f *FakePNL) CalculatePNL(context.Context, *PNLCalculatorRequest) (*PNLResult, error) {
|
|
if f.err != nil {
|
|
return nil, f.err
|
|
}
|
|
return f.result, nil
|
|
}
|
|
|
|
// GetCurrencyForRealisedPNL overrides default pnl calculations
|
|
func (f *FakePNL) GetCurrencyForRealisedPNL(realisedAsset asset.Item, realisedPair currency.Pair) (currency.Code, asset.Item, error) {
|
|
if f.err != nil {
|
|
return realisedPair.Base, asset.Empty, f.err
|
|
}
|
|
return realisedPair.Base, realisedAsset, nil
|
|
}
|
|
|
|
func TestUpsertPNLEntry(t *testing.T) {
|
|
t.Parallel()
|
|
var results []PNLResult
|
|
result := &PNLResult{
|
|
IsOrder: true,
|
|
}
|
|
_, err := upsertPNLEntry(results, result)
|
|
if !errors.Is(err, errTimeUnset) {
|
|
t.Error(err)
|
|
}
|
|
tt := time.Now()
|
|
result.Time = tt
|
|
results, err = upsertPNLEntry(results, result)
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
if len(results) != 1 {
|
|
t.Errorf("expected 1 received %v", len(results))
|
|
}
|
|
result.Fee = decimal.NewFromInt(1337)
|
|
results, err = upsertPNLEntry(results, result)
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
if len(results) != 1 {
|
|
t.Errorf("expected 1 received %v", len(results))
|
|
}
|
|
if !results[0].Fee.Equal(result.Fee) {
|
|
t.Errorf("expected %v received %v", result.Fee, results[0].Fee)
|
|
}
|
|
}
|
|
|
|
func TestTrackNewOrder(t *testing.T) {
|
|
t.Parallel()
|
|
exch := testExchange
|
|
item := asset.Futures
|
|
pair, err := currency.NewPairFromStrings("BTC", "1231")
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
e := MultiPositionTracker{
|
|
exchange: testExchange,
|
|
exchangePNLCalculation: &FakePNL{},
|
|
}
|
|
setup := &PositionTrackerSetup{
|
|
Pair: pair,
|
|
Asset: item,
|
|
}
|
|
f, err := e.SetupPositionTracker(setup)
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
|
|
err = f.TrackNewOrder(nil, false)
|
|
if !errors.Is(err, ErrSubmissionIsNil) {
|
|
t.Error(err)
|
|
}
|
|
err = f.TrackNewOrder(&Detail{}, false)
|
|
if !errors.Is(err, errOrderNotEqualToTracker) {
|
|
t.Error(err)
|
|
}
|
|
|
|
od := &Detail{
|
|
Exchange: exch,
|
|
AssetType: item,
|
|
Pair: pair,
|
|
OrderID: "1",
|
|
Price: 1337,
|
|
}
|
|
err = f.TrackNewOrder(od, false)
|
|
if !errors.Is(err, ErrSideIsInvalid) {
|
|
t.Error(err)
|
|
}
|
|
|
|
od.Side = Long
|
|
od.Amount = 1
|
|
od.OrderID = "2"
|
|
err = f.TrackNewOrder(od, false)
|
|
if !errors.Is(err, errTimeUnset) {
|
|
t.Error(err)
|
|
}
|
|
f.openingDirection = Long
|
|
od.Date = time.Now()
|
|
err = f.TrackNewOrder(od, false)
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
if !f.entryPrice.Equal(decimal.NewFromInt(1337)) {
|
|
t.Errorf("expected 1337, received %v", f.entryPrice)
|
|
}
|
|
if len(f.longPositions) != 1 {
|
|
t.Error("expected a long")
|
|
}
|
|
if f.currentDirection != Long {
|
|
t.Error("expected recognition that its long")
|
|
}
|
|
if f.exposure.InexactFloat64() != od.Amount {
|
|
t.Error("expected 1")
|
|
}
|
|
|
|
od.Date = od.Date.Add(1)
|
|
od.Amount = 0.4
|
|
od.Side = Short
|
|
od.OrderID = "3"
|
|
err = f.TrackNewOrder(od, false)
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
if len(f.shortPositions) != 1 {
|
|
t.Error("expected a short")
|
|
}
|
|
if f.currentDirection != Long {
|
|
t.Error("expected recognition that its long")
|
|
}
|
|
if f.exposure.InexactFloat64() != 0.6 {
|
|
t.Error("expected 0.6")
|
|
}
|
|
|
|
od.Date = od.Date.Add(1)
|
|
od.Amount = 0.8
|
|
od.Side = Short
|
|
od.OrderID = "4"
|
|
od.Fee = 0.1
|
|
err = f.TrackNewOrder(od, false)
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
if f.currentDirection != Short {
|
|
t.Error("expected recognition that its short")
|
|
}
|
|
if !f.exposure.Equal(decimal.NewFromFloat(0.2)) {
|
|
t.Errorf("expected %v received %v", 0.2, f.exposure)
|
|
}
|
|
|
|
od.Date = od.Date.Add(1)
|
|
od.OrderID = "5"
|
|
od.Side = Long
|
|
od.Amount = 0.2
|
|
err = f.TrackNewOrder(od, false)
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
if f.currentDirection != ClosePosition {
|
|
t.Errorf("expected recognition that its unknown, received '%v'", f.currentDirection)
|
|
}
|
|
if f.status != Closed {
|
|
t.Errorf("expected recognition that its closed, received '%v'", f.status)
|
|
}
|
|
|
|
err = f.TrackNewOrder(od, false)
|
|
if !errors.Is(err, ErrPositionClosed) {
|
|
t.Error(err)
|
|
}
|
|
if f.currentDirection != ClosePosition {
|
|
t.Errorf("expected recognition that its unknown, received '%v'", f.currentDirection)
|
|
}
|
|
if f.status != Closed {
|
|
t.Errorf("expected recognition that its closed, received '%v'", f.status)
|
|
}
|
|
|
|
err = f.TrackNewOrder(od, true)
|
|
if !errors.Is(err, errCannotTrackInvalidParams) {
|
|
t.Error(err)
|
|
}
|
|
|
|
f, err = e.SetupPositionTracker(setup)
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
err = f.TrackNewOrder(od, true)
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
}
|
|
|
|
func TestSetupMultiPositionTracker(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
_, err := SetupMultiPositionTracker(nil)
|
|
if !errors.Is(err, errNilSetup) {
|
|
t.Error(err)
|
|
}
|
|
|
|
setup := &MultiPositionTrackerSetup{}
|
|
_, err = SetupMultiPositionTracker(setup)
|
|
if !errors.Is(err, errExchangeNameEmpty) {
|
|
t.Error(err)
|
|
}
|
|
setup.Exchange = testExchange
|
|
_, err = SetupMultiPositionTracker(setup)
|
|
if !errors.Is(err, ErrNotFuturesAsset) {
|
|
t.Error(err)
|
|
}
|
|
setup.Asset = asset.Futures
|
|
_, err = SetupMultiPositionTracker(setup)
|
|
if !errors.Is(err, ErrPairIsEmpty) {
|
|
t.Error(err)
|
|
}
|
|
|
|
setup.Pair = currency.NewPair(currency.BTC, currency.USDT)
|
|
_, err = SetupMultiPositionTracker(setup)
|
|
if !errors.Is(err, errEmptyUnderlying) {
|
|
t.Error(err)
|
|
}
|
|
|
|
setup.Underlying = currency.BTC
|
|
_, err = SetupMultiPositionTracker(setup)
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
|
|
setup.UseExchangePNLCalculation = true
|
|
_, err = SetupMultiPositionTracker(setup)
|
|
if !errors.Is(err, errMissingPNLCalculationFunctions) {
|
|
t.Error(err)
|
|
}
|
|
|
|
setup.ExchangePNLCalculation = &FakePNL{}
|
|
resp, err := SetupMultiPositionTracker(setup)
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
if resp.exchange != testExchange {
|
|
t.Errorf("expected 'test' received %v", resp.exchange)
|
|
}
|
|
}
|
|
|
|
func TestExchangeTrackNewOrder(t *testing.T) {
|
|
t.Parallel()
|
|
exch := testExchange
|
|
item := asset.Futures
|
|
pair := currency.NewPair(currency.BTC, currency.USDT)
|
|
setup := &MultiPositionTrackerSetup{
|
|
Exchange: exch,
|
|
Asset: item,
|
|
Pair: pair,
|
|
Underlying: pair.Base,
|
|
ExchangePNLCalculation: &FakePNL{},
|
|
}
|
|
resp, err := SetupMultiPositionTracker(setup)
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
|
|
tt := time.Now()
|
|
|
|
err = resp.TrackNewOrder(&Detail{
|
|
Date: tt,
|
|
Exchange: exch,
|
|
AssetType: item,
|
|
Pair: pair,
|
|
Side: Short,
|
|
OrderID: "1",
|
|
Amount: 1,
|
|
})
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
if len(resp.positions) != 1 {
|
|
t.Errorf("expected '1' received %v", len(resp.positions))
|
|
}
|
|
|
|
err = resp.TrackNewOrder(&Detail{
|
|
Date: tt,
|
|
Exchange: exch,
|
|
AssetType: item,
|
|
Pair: pair,
|
|
Side: Short,
|
|
OrderID: "2",
|
|
Amount: 1,
|
|
})
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
if len(resp.positions) != 1 {
|
|
t.Errorf("expected '1' received %v", len(resp.positions))
|
|
}
|
|
|
|
err = resp.TrackNewOrder(&Detail{
|
|
Date: tt,
|
|
Exchange: exch,
|
|
AssetType: item,
|
|
Pair: pair,
|
|
Side: Long,
|
|
OrderID: "3",
|
|
Amount: 2,
|
|
})
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
if len(resp.positions) != 1 {
|
|
t.Errorf("expected '1' received %v", len(resp.positions))
|
|
}
|
|
if resp.positions[0].status != Closed {
|
|
t.Errorf("expected 'closed' received %v", resp.positions[0].status)
|
|
}
|
|
resp.positions[0].status = Open
|
|
resp.positions = append(resp.positions, resp.positions...)
|
|
err = resp.TrackNewOrder(&Detail{
|
|
Date: tt,
|
|
Exchange: exch,
|
|
AssetType: item,
|
|
Pair: pair,
|
|
Side: Long,
|
|
OrderID: "4",
|
|
Amount: 2,
|
|
})
|
|
if !errors.Is(err, errPositionDiscrepancy) {
|
|
t.Errorf("received '%v' expected '%v", err, errPositionDiscrepancy)
|
|
}
|
|
|
|
resp.positions = []*PositionTracker{resp.positions[0]}
|
|
resp.positions[0].status = Closed
|
|
err = resp.TrackNewOrder(&Detail{
|
|
Date: tt,
|
|
Exchange: exch,
|
|
AssetType: item,
|
|
Pair: pair,
|
|
Side: Long,
|
|
OrderID: "4",
|
|
Amount: 2,
|
|
})
|
|
if !errors.Is(err, nil) {
|
|
t.Errorf("received '%v' expected '%v", err, nil)
|
|
}
|
|
if len(resp.positions) != 2 {
|
|
t.Errorf("expected '2' received %v", len(resp.positions))
|
|
}
|
|
|
|
resp.positions[0].status = Closed
|
|
err = resp.TrackNewOrder(&Detail{
|
|
Date: tt,
|
|
Exchange: exch,
|
|
Pair: pair,
|
|
AssetType: asset.USDTMarginedFutures,
|
|
Side: Long,
|
|
OrderID: "5",
|
|
Amount: 2,
|
|
})
|
|
if !errors.Is(err, errAssetMismatch) {
|
|
t.Error(err)
|
|
}
|
|
}
|
|
|
|
func TestSetupPositionControllerReal(t *testing.T) {
|
|
t.Parallel()
|
|
pc := SetupPositionController()
|
|
if pc.multiPositionTrackers == nil {
|
|
t.Error("unexpected nil")
|
|
}
|
|
}
|
|
|
|
func TestPositionControllerTestTrackNewOrder(t *testing.T) {
|
|
t.Parallel()
|
|
pc := SetupPositionController()
|
|
err := pc.TrackNewOrder(nil)
|
|
if !errors.Is(err, errNilOrder) {
|
|
t.Error(err)
|
|
}
|
|
|
|
err = pc.TrackNewOrder(&Detail{
|
|
Date: time.Now(),
|
|
Exchange: "hi",
|
|
Pair: currency.NewPair(currency.BTC, currency.USDT),
|
|
AssetType: asset.Spot,
|
|
Side: Long,
|
|
OrderID: "lol",
|
|
})
|
|
if !errors.Is(err, ErrNotFuturesAsset) {
|
|
t.Error(err)
|
|
}
|
|
|
|
err = pc.TrackNewOrder(&Detail{
|
|
Date: time.Now(),
|
|
Exchange: "hi",
|
|
Pair: currency.NewPair(currency.BTC, currency.USDT),
|
|
AssetType: asset.Futures,
|
|
Side: Long,
|
|
OrderID: "lol",
|
|
})
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
}
|
|
|
|
func TestGetLatestPNLSnapshot(t *testing.T) {
|
|
t.Parallel()
|
|
pt := PositionTracker{}
|
|
_, err := pt.GetLatestPNLSnapshot()
|
|
if !errors.Is(err, errNoPNLHistory) {
|
|
t.Error(err)
|
|
}
|
|
|
|
pnl := PNLResult{
|
|
Time: time.Now(),
|
|
UnrealisedPNL: decimal.NewFromInt(1337),
|
|
RealisedPNLBeforeFees: decimal.NewFromInt(1337),
|
|
}
|
|
pt.pnlHistory = append(pt.pnlHistory, pnl)
|
|
|
|
result, err := pt.GetLatestPNLSnapshot()
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
if result != pt.pnlHistory[0] {
|
|
t.Error("unexpected result")
|
|
}
|
|
}
|
|
|
|
func TestGetRealisedPNL(t *testing.T) {
|
|
t.Parallel()
|
|
p := PositionTracker{}
|
|
result := p.GetRealisedPNL()
|
|
if !result.IsZero() {
|
|
t.Error("expected zero")
|
|
}
|
|
}
|
|
|
|
func TestGetStats(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
p := &PositionTracker{}
|
|
stats := p.GetStats()
|
|
if len(stats.Orders) != 0 {
|
|
t.Error("expected 0")
|
|
}
|
|
|
|
p.exchange = testExchange
|
|
stats = p.GetStats()
|
|
if stats.Exchange != p.exchange {
|
|
t.Errorf("expected '%v' received '%v'", p.exchange, stats.Exchange)
|
|
}
|
|
|
|
p = nil
|
|
stats = p.GetStats()
|
|
if len(stats.Orders) != 0 {
|
|
t.Error("expected 0")
|
|
}
|
|
}
|
|
|
|
func TestGetPositions(t *testing.T) {
|
|
t.Parallel()
|
|
p := &MultiPositionTracker{}
|
|
positions := p.GetPositions()
|
|
if len(positions) > 0 {
|
|
t.Error("expected 0")
|
|
}
|
|
|
|
p.positions = append(p.positions, &PositionTracker{
|
|
exchange: testExchange,
|
|
})
|
|
positions = p.GetPositions()
|
|
if len(positions) != 1 {
|
|
t.Fatal("expected 1")
|
|
}
|
|
if positions[0].Exchange != testExchange {
|
|
t.Error("expected 'test'")
|
|
}
|
|
|
|
p = nil
|
|
positions = p.GetPositions()
|
|
if len(positions) > 0 {
|
|
t.Error("expected 0")
|
|
}
|
|
}
|
|
|
|
func TestGetPositionsForExchange(t *testing.T) {
|
|
t.Parallel()
|
|
c := &PositionController{}
|
|
p := currency.NewPair(currency.BTC, currency.USDT)
|
|
pos, err := c.GetPositionsForExchange(testExchange, asset.Futures, p)
|
|
if !errors.Is(err, ErrPositionsNotLoadedForExchange) {
|
|
t.Errorf("received '%v' expected '%v", err, ErrPositionsNotLoadedForExchange)
|
|
}
|
|
if len(pos) != 0 {
|
|
t.Error("expected zero")
|
|
}
|
|
c.multiPositionTrackers = make(map[string]map[asset.Item]map[currency.Pair]*MultiPositionTracker)
|
|
c.multiPositionTrackers[testExchange] = nil
|
|
_, err = c.GetPositionsForExchange(testExchange, asset.Futures, p)
|
|
if !errors.Is(err, ErrPositionsNotLoadedForAsset) {
|
|
t.Errorf("received '%v' expected '%v", err, ErrPositionsNotLoadedForExchange)
|
|
}
|
|
c.multiPositionTrackers[testExchange] = make(map[asset.Item]map[currency.Pair]*MultiPositionTracker)
|
|
c.multiPositionTrackers[testExchange][asset.Futures] = nil
|
|
_, err = c.GetPositionsForExchange(testExchange, asset.Futures, p)
|
|
if !errors.Is(err, ErrPositionsNotLoadedForPair) {
|
|
t.Errorf("received '%v' expected '%v", err, ErrPositionsNotLoadedForPair)
|
|
}
|
|
_, err = c.GetPositionsForExchange(testExchange, asset.Spot, p)
|
|
if !errors.Is(err, ErrNotFuturesAsset) {
|
|
t.Errorf("received '%v' expected '%v", err, ErrNotFuturesAsset)
|
|
}
|
|
|
|
c.multiPositionTrackers[testExchange][asset.Futures] = make(map[currency.Pair]*MultiPositionTracker)
|
|
c.multiPositionTrackers[testExchange][asset.Futures][p] = &MultiPositionTracker{
|
|
exchange: testExchange,
|
|
}
|
|
|
|
pos, err = c.GetPositionsForExchange(testExchange, asset.Futures, p)
|
|
if !errors.Is(err, nil) {
|
|
t.Errorf("received '%v' expected '%v", err, nil)
|
|
}
|
|
if len(pos) != 0 {
|
|
t.Fatal("expected zero")
|
|
}
|
|
c.multiPositionTrackers[testExchange][asset.Futures][p] = &MultiPositionTracker{
|
|
exchange: testExchange,
|
|
positions: []*PositionTracker{
|
|
{
|
|
exchange: testExchange,
|
|
},
|
|
},
|
|
}
|
|
pos, err = c.GetPositionsForExchange(testExchange, asset.Futures, p)
|
|
if !errors.Is(err, nil) {
|
|
t.Errorf("received '%v' expected '%v", err, nil)
|
|
}
|
|
if len(pos) != 1 {
|
|
t.Fatal("expected 1")
|
|
}
|
|
if pos[0].Exchange != testExchange {
|
|
t.Error("expected test")
|
|
}
|
|
c = nil
|
|
_, err = c.GetPositionsForExchange(testExchange, asset.Futures, p)
|
|
if !errors.Is(err, common.ErrNilPointer) {
|
|
t.Errorf("received '%v' expected '%v", err, common.ErrNilPointer)
|
|
}
|
|
}
|
|
|
|
func TestClearPositionsForExchange(t *testing.T) {
|
|
t.Parallel()
|
|
c := &PositionController{}
|
|
p := currency.NewPair(currency.BTC, currency.USDT)
|
|
err := c.ClearPositionsForExchange(testExchange, asset.Futures, p)
|
|
if !errors.Is(err, ErrPositionsNotLoadedForExchange) {
|
|
t.Errorf("received '%v' expected '%v", err, ErrPositionsNotLoadedForExchange)
|
|
}
|
|
c.multiPositionTrackers = make(map[string]map[asset.Item]map[currency.Pair]*MultiPositionTracker)
|
|
c.multiPositionTrackers[testExchange] = nil
|
|
err = c.ClearPositionsForExchange(testExchange, asset.Futures, p)
|
|
if !errors.Is(err, ErrPositionsNotLoadedForAsset) {
|
|
t.Errorf("received '%v' expected '%v", err, ErrPositionsNotLoadedForExchange)
|
|
}
|
|
c.multiPositionTrackers[testExchange] = make(map[asset.Item]map[currency.Pair]*MultiPositionTracker)
|
|
c.multiPositionTrackers[testExchange][asset.Futures] = nil
|
|
err = c.ClearPositionsForExchange(testExchange, asset.Futures, p)
|
|
if !errors.Is(err, ErrPositionsNotLoadedForPair) {
|
|
t.Errorf("received '%v' expected '%v", err, ErrPositionsNotLoadedForPair)
|
|
}
|
|
err = c.ClearPositionsForExchange(testExchange, asset.Spot, p)
|
|
if !errors.Is(err, ErrNotFuturesAsset) {
|
|
t.Errorf("received '%v' expected '%v", err, ErrNotFuturesAsset)
|
|
}
|
|
|
|
c.multiPositionTrackers[testExchange][asset.Futures] = make(map[currency.Pair]*MultiPositionTracker)
|
|
c.multiPositionTrackers[testExchange][asset.Futures][p] = &MultiPositionTracker{
|
|
exchange: testExchange,
|
|
}
|
|
c.multiPositionTrackers[testExchange][asset.Futures][p] = &MultiPositionTracker{
|
|
exchange: testExchange,
|
|
underlying: currency.DOGE,
|
|
positions: []*PositionTracker{
|
|
{
|
|
exchange: testExchange,
|
|
},
|
|
},
|
|
}
|
|
err = c.ClearPositionsForExchange(testExchange, asset.Futures, p)
|
|
if !errors.Is(err, nil) {
|
|
t.Errorf("received '%v' expected '%v", err, nil)
|
|
}
|
|
if len(c.multiPositionTrackers[testExchange][asset.Futures][p].positions) != 0 {
|
|
t.Fatal("expected 0")
|
|
}
|
|
c = nil
|
|
_, err = c.GetPositionsForExchange(testExchange, asset.Futures, p)
|
|
if !errors.Is(err, common.ErrNilPointer) {
|
|
t.Errorf("received '%v' expected '%v", err, common.ErrNilPointer)
|
|
}
|
|
}
|
|
|
|
func TestCalculateRealisedPNL(t *testing.T) {
|
|
t.Parallel()
|
|
result := calculateRealisedPNL(nil)
|
|
if !result.IsZero() {
|
|
t.Errorf("received '%v' expected '0'", result)
|
|
}
|
|
result = calculateRealisedPNL([]PNLResult{
|
|
{
|
|
IsOrder: true,
|
|
RealisedPNLBeforeFees: decimal.NewFromInt(1337),
|
|
},
|
|
})
|
|
if !result.Equal(decimal.NewFromInt(1337)) {
|
|
t.Errorf("received '%v' expected '1337'", result)
|
|
}
|
|
|
|
result = calculateRealisedPNL([]PNLResult{
|
|
{
|
|
IsOrder: true,
|
|
RealisedPNLBeforeFees: decimal.NewFromInt(1339),
|
|
Fee: decimal.NewFromInt(2),
|
|
},
|
|
{
|
|
IsOrder: true,
|
|
RealisedPNLBeforeFees: decimal.NewFromInt(2),
|
|
Fee: decimal.NewFromInt(2),
|
|
},
|
|
})
|
|
if !result.Equal(decimal.NewFromInt(1337)) {
|
|
t.Errorf("received '%v' expected '1337'", result)
|
|
}
|
|
}
|
|
|
|
func TestSetupPositionTracker(t *testing.T) {
|
|
t.Parallel()
|
|
m := &MultiPositionTracker{}
|
|
p, err := m.SetupPositionTracker(nil)
|
|
if !errors.Is(err, errExchangeNameEmpty) {
|
|
t.Errorf("received '%v' expected '%v", err, errExchangeNameEmpty)
|
|
}
|
|
if p != nil {
|
|
t.Error("expected nil")
|
|
}
|
|
m.exchange = testExchange
|
|
p, err = m.SetupPositionTracker(nil)
|
|
if !errors.Is(err, errNilSetup) {
|
|
t.Errorf("received '%v' expected '%v", err, errNilSetup)
|
|
}
|
|
if p != nil {
|
|
t.Error("expected nil")
|
|
}
|
|
|
|
p, err = m.SetupPositionTracker(&PositionTrackerSetup{
|
|
Asset: asset.Spot,
|
|
})
|
|
if !errors.Is(err, ErrNotFuturesAsset) {
|
|
t.Errorf("received '%v' expected '%v", err, ErrNotFuturesAsset)
|
|
}
|
|
if p != nil {
|
|
t.Error("expected nil")
|
|
}
|
|
|
|
p, err = m.SetupPositionTracker(&PositionTrackerSetup{
|
|
Asset: asset.Futures,
|
|
})
|
|
if !errors.Is(err, ErrPairIsEmpty) {
|
|
t.Errorf("received '%v' expected '%v", err, ErrPairIsEmpty)
|
|
}
|
|
if p != nil {
|
|
t.Error("expected nil")
|
|
}
|
|
|
|
cp := currency.NewPair(currency.BTC, currency.USDT)
|
|
p, err = m.SetupPositionTracker(&PositionTrackerSetup{
|
|
Asset: asset.Futures,
|
|
Pair: cp,
|
|
})
|
|
if !errors.Is(err, nil) {
|
|
t.Fatalf("received '%v' expected '%v", err, nil)
|
|
}
|
|
if p == nil { //nolint:staticcheck,nolintlint // SA5011 Ignore the nil warnings
|
|
t.Fatal("expected not nil")
|
|
}
|
|
if p.exchange != testExchange { //nolint:staticcheck,nolintlint // SA5011 Ignore the nil warnings
|
|
t.Error("expected test")
|
|
}
|
|
|
|
_, err = m.SetupPositionTracker(&PositionTrackerSetup{
|
|
Asset: asset.Futures,
|
|
Pair: cp,
|
|
UseExchangePNLCalculation: true,
|
|
})
|
|
if !errors.Is(err, ErrNilPNLCalculator) {
|
|
t.Errorf("received '%v' expected '%v", err, ErrNilPNLCalculator)
|
|
}
|
|
m.exchangePNLCalculation = &PNLCalculator{}
|
|
p, err = m.SetupPositionTracker(&PositionTrackerSetup{
|
|
Asset: asset.Futures,
|
|
Pair: cp,
|
|
UseExchangePNLCalculation: true,
|
|
})
|
|
if !errors.Is(err, nil) {
|
|
t.Errorf("received '%v' expected '%v", err, nil)
|
|
}
|
|
if !p.useExchangePNLCalculation {
|
|
t.Error("expected true")
|
|
}
|
|
}
|
|
|
|
func TestCalculatePNL(t *testing.T) {
|
|
t.Parallel()
|
|
p := &PNLCalculator{}
|
|
_, err := p.CalculatePNL(context.Background(), nil)
|
|
if !errors.Is(err, ErrNilPNLCalculator) {
|
|
t.Errorf("received '%v' expected '%v", err, ErrNilPNLCalculator)
|
|
}
|
|
_, err = p.CalculatePNL(context.Background(), &PNLCalculatorRequest{})
|
|
if !errors.Is(err, errCannotCalculateUnrealisedPNL) {
|
|
t.Errorf("received '%v' expected '%v", err, errCannotCalculateUnrealisedPNL)
|
|
}
|
|
|
|
_, err = p.CalculatePNL(context.Background(),
|
|
&PNLCalculatorRequest{
|
|
OrderDirection: Short,
|
|
CurrentDirection: Long,
|
|
})
|
|
if !errors.Is(err, errCannotCalculateUnrealisedPNL) {
|
|
t.Errorf("received '%v' expected '%v", err, errCannotCalculateUnrealisedPNL)
|
|
}
|
|
}
|
|
|
|
func TestTrackPNLByTime(t *testing.T) {
|
|
t.Parallel()
|
|
p := &PositionTracker{}
|
|
err := p.TrackPNLByTime(time.Now(), 1)
|
|
if !errors.Is(err, nil) {
|
|
t.Errorf("received '%v' expected '%v", err, nil)
|
|
}
|
|
|
|
err = p.TrackPNLByTime(time.Now(), 2)
|
|
if !errors.Is(err, nil) {
|
|
t.Errorf("received '%v' expected '%v", err, nil)
|
|
}
|
|
if !p.latestPrice.Equal(decimal.NewFromInt(2)) {
|
|
t.Error("expected 2")
|
|
}
|
|
p = nil
|
|
err = p.TrackPNLByTime(time.Now(), 2)
|
|
if !errors.Is(err, common.ErrNilPointer) {
|
|
t.Errorf("received '%v' expected '%v", err, common.ErrNilPointer)
|
|
}
|
|
}
|
|
|
|
func TestUpdateOpenPositionUnrealisedPNL(t *testing.T) {
|
|
t.Parallel()
|
|
pc := SetupPositionController()
|
|
|
|
_, err := pc.UpdateOpenPositionUnrealisedPNL("hi", asset.Futures, currency.NewPair(currency.BTC, currency.USDT), 2, time.Now())
|
|
if !errors.Is(err, ErrPositionsNotLoadedForExchange) {
|
|
t.Errorf("received '%v' expected '%v", err, ErrPositionsNotLoadedForExchange)
|
|
}
|
|
|
|
_, err = pc.UpdateOpenPositionUnrealisedPNL("hi", asset.Spot, currency.NewPair(currency.BTC, currency.USDT), 2, time.Now())
|
|
if !errors.Is(err, ErrNotFuturesAsset) {
|
|
t.Errorf("received '%v' expected '%v", err, ErrNotFuturesAsset)
|
|
}
|
|
|
|
err = pc.TrackNewOrder(&Detail{
|
|
Date: time.Now(),
|
|
Exchange: "hi",
|
|
Pair: currency.NewPair(currency.BTC, currency.USDT),
|
|
AssetType: asset.Futures,
|
|
Side: Long,
|
|
OrderID: "lol",
|
|
Price: 1,
|
|
Amount: 1,
|
|
})
|
|
if !errors.Is(err, nil) {
|
|
t.Errorf("received '%v' expected '%v", err, nil)
|
|
}
|
|
|
|
_, err = pc.UpdateOpenPositionUnrealisedPNL("hi2", asset.Futures, currency.NewPair(currency.BTC, currency.USDT), 2, time.Now())
|
|
if !errors.Is(err, ErrPositionsNotLoadedForExchange) {
|
|
t.Errorf("received '%v' expected '%v", err, ErrPositionsNotLoadedForExchange)
|
|
}
|
|
|
|
_, err = pc.UpdateOpenPositionUnrealisedPNL("hi", asset.PerpetualSwap, currency.NewPair(currency.BTC, currency.USDT), 2, time.Now())
|
|
if !errors.Is(err, ErrPositionsNotLoadedForAsset) {
|
|
t.Errorf("received '%v' expected '%v", err, ErrPositionsNotLoadedForAsset)
|
|
}
|
|
|
|
_, err = pc.UpdateOpenPositionUnrealisedPNL("hi", asset.Futures, currency.NewPair(currency.BTC, currency.DOGE), 2, time.Now())
|
|
if !errors.Is(err, ErrPositionsNotLoadedForPair) {
|
|
t.Errorf("received '%v' expected '%v", err, ErrPositionsNotLoadedForPair)
|
|
}
|
|
|
|
pnl, err := pc.UpdateOpenPositionUnrealisedPNL("hi", asset.Futures, currency.NewPair(currency.BTC, currency.USDT), 2, time.Now())
|
|
if !errors.Is(err, nil) {
|
|
t.Errorf("received '%v' expected '%v", err, nil)
|
|
}
|
|
if !pnl.Equal(decimal.NewFromInt(1)) {
|
|
t.Errorf("received '%v' expected '%v", pnl, 1)
|
|
}
|
|
|
|
pc = nil
|
|
_, err = pc.UpdateOpenPositionUnrealisedPNL("hi", asset.Futures, currency.NewPair(currency.BTC, currency.USDT), 2, time.Now())
|
|
if !errors.Is(err, common.ErrNilPointer) {
|
|
t.Errorf("received '%v' expected '%v", err, common.ErrNilPointer)
|
|
}
|
|
}
|
|
|
|
func TestSetCollateralCurrency(t *testing.T) {
|
|
t.Parallel()
|
|
var expectedError = ErrNotFuturesAsset
|
|
pc := SetupPositionController()
|
|
err := pc.SetCollateralCurrency("hi", asset.Spot, currency.Pair{}, currency.Code{})
|
|
if !errors.Is(err, expectedError) {
|
|
t.Errorf("received '%v' expected '%v", err, expectedError)
|
|
}
|
|
cp := currency.NewPair(currency.BTC, currency.USDT)
|
|
pc.multiPositionTrackers = make(map[string]map[asset.Item]map[currency.Pair]*MultiPositionTracker)
|
|
err = pc.SetCollateralCurrency("hi", asset.Futures, cp, currency.DOGE)
|
|
expectedError = ErrPositionsNotLoadedForExchange
|
|
if !errors.Is(err, expectedError) {
|
|
t.Fatalf("received '%v' expected '%v", err, expectedError)
|
|
}
|
|
pc.multiPositionTrackers["hi"] = make(map[asset.Item]map[currency.Pair]*MultiPositionTracker)
|
|
err = pc.SetCollateralCurrency("hi", asset.Futures, cp, currency.DOGE)
|
|
expectedError = ErrPositionsNotLoadedForAsset
|
|
if !errors.Is(err, expectedError) {
|
|
t.Fatalf("received '%v' expected '%v", err, expectedError)
|
|
}
|
|
|
|
pc.multiPositionTrackers["hi"][asset.Futures] = make(map[currency.Pair]*MultiPositionTracker)
|
|
err = pc.SetCollateralCurrency("hi", asset.Futures, cp, currency.DOGE)
|
|
expectedError = ErrPositionsNotLoadedForPair
|
|
if !errors.Is(err, expectedError) {
|
|
t.Fatalf("received '%v' expected '%v", err, expectedError)
|
|
}
|
|
|
|
pc.multiPositionTrackers["hi"][asset.Futures][cp] = nil
|
|
err = pc.SetCollateralCurrency("hi", asset.Futures, cp, currency.DOGE)
|
|
expectedError = common.ErrNilPointer
|
|
if !errors.Is(err, expectedError) {
|
|
t.Fatalf("received '%v' expected '%v", err, expectedError)
|
|
}
|
|
|
|
pc.multiPositionTrackers["hi"][asset.Futures][cp] = &MultiPositionTracker{
|
|
exchange: "hi",
|
|
asset: asset.Futures,
|
|
pair: cp,
|
|
orderPositions: make(map[string]*PositionTracker),
|
|
}
|
|
err = pc.TrackNewOrder(&Detail{
|
|
Date: time.Now(),
|
|
Exchange: "hi",
|
|
Pair: cp,
|
|
AssetType: asset.Futures,
|
|
Side: Long,
|
|
OrderID: "lol",
|
|
Price: 1,
|
|
Amount: 1,
|
|
})
|
|
if !errors.Is(err, nil) {
|
|
t.Fatalf("received '%v' expected '%v", err, nil)
|
|
}
|
|
|
|
err = pc.SetCollateralCurrency("hi", asset.Futures, cp, currency.DOGE)
|
|
expectedError = nil
|
|
if !errors.Is(err, expectedError) {
|
|
t.Fatalf("received '%v' expected '%v", err, expectedError)
|
|
}
|
|
|
|
if !pc.multiPositionTrackers["hi"][asset.Futures][cp].collateralCurrency.Equal(currency.DOGE) {
|
|
t.Errorf("received '%v' expected '%v'", pc.multiPositionTrackers["hi"][asset.Futures][cp].collateralCurrency, currency.DOGE)
|
|
}
|
|
|
|
if !pc.multiPositionTrackers["hi"][asset.Futures][cp].positions[0].collateralCurrency.Equal(currency.DOGE) {
|
|
t.Errorf("received '%v' expected '%v'", pc.multiPositionTrackers["hi"][asset.Futures][cp].positions[0].collateralCurrency, currency.DOGE)
|
|
}
|
|
|
|
pc = nil
|
|
err = pc.SetCollateralCurrency("hi", asset.Spot, currency.Pair{}, currency.Code{})
|
|
expectedError = common.ErrNilPointer
|
|
if !errors.Is(err, expectedError) {
|
|
t.Errorf("received '%v' expected '%v", err, expectedError)
|
|
}
|
|
}
|
|
|
|
func TestMPTUpdateOpenPositionUnrealisedPNL(t *testing.T) {
|
|
t.Parallel()
|
|
var err, expectedError error
|
|
expectedError = nil
|
|
cp := currency.NewPair(currency.BTC, currency.USDT)
|
|
pc := SetupPositionController()
|
|
err = pc.TrackNewOrder(&Detail{
|
|
Date: time.Now(),
|
|
Exchange: "hi",
|
|
Pair: cp,
|
|
AssetType: asset.Futures,
|
|
Side: Long,
|
|
OrderID: "lol",
|
|
Price: 1,
|
|
Amount: 1,
|
|
})
|
|
if !errors.Is(err, expectedError) {
|
|
t.Fatalf("received '%v' expected '%v", err, expectedError)
|
|
}
|
|
result, err := pc.multiPositionTrackers["hi"][asset.Futures][cp].UpdateOpenPositionUnrealisedPNL(1337, time.Now())
|
|
if !errors.Is(err, expectedError) {
|
|
t.Fatalf("received '%v' expected '%v", err, expectedError)
|
|
}
|
|
if result.Equal(decimal.NewFromInt(1337)) {
|
|
t.Error("")
|
|
}
|
|
|
|
expectedError = ErrPositionClosed
|
|
pc.multiPositionTrackers["hi"][asset.Futures][cp].positions[0].status = Closed
|
|
_, err = pc.multiPositionTrackers["hi"][asset.Futures][cp].UpdateOpenPositionUnrealisedPNL(1337, time.Now())
|
|
if !errors.Is(err, expectedError) {
|
|
t.Fatalf("received '%v' expected '%v", err, expectedError)
|
|
}
|
|
|
|
expectedError = ErrPositionsNotLoadedForPair
|
|
pc.multiPositionTrackers["hi"][asset.Futures][cp].positions = nil
|
|
_, err = pc.multiPositionTrackers["hi"][asset.Futures][cp].UpdateOpenPositionUnrealisedPNL(1337, time.Now())
|
|
if !errors.Is(err, expectedError) {
|
|
t.Fatalf("received '%v' expected '%v", err, expectedError)
|
|
}
|
|
}
|
|
|
|
func TestMPTLiquidate(t *testing.T) {
|
|
t.Parallel()
|
|
item := asset.Futures
|
|
pair, err := currency.NewPairFromStrings("BTC", "1231")
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
e := &MultiPositionTracker{
|
|
exchange: testExchange,
|
|
exchangePNLCalculation: &FakePNL{},
|
|
asset: item,
|
|
orderPositions: make(map[string]*PositionTracker),
|
|
}
|
|
|
|
err = e.Liquidate(decimal.Zero, time.Time{})
|
|
if !errors.Is(err, ErrPositionsNotLoadedForPair) {
|
|
t.Error(err)
|
|
}
|
|
|
|
setup := &PositionTrackerSetup{
|
|
Pair: pair,
|
|
Asset: item,
|
|
}
|
|
_, err = e.SetupPositionTracker(setup)
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
|
|
tt := time.Now()
|
|
err = e.TrackNewOrder(&Detail{
|
|
Date: tt,
|
|
Exchange: testExchange,
|
|
Pair: pair,
|
|
AssetType: item,
|
|
Side: Long,
|
|
OrderID: "lol",
|
|
Price: 1,
|
|
Amount: 1,
|
|
})
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
|
|
err = e.Liquidate(decimal.Zero, time.Time{})
|
|
if !errors.Is(err, errCannotLiquidate) {
|
|
t.Error(err)
|
|
}
|
|
|
|
err = e.Liquidate(decimal.Zero, tt)
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
|
|
if e.positions[0].status != Liquidated {
|
|
t.Errorf("received '%v' expected '%v'", e.positions[0].status, Liquidated)
|
|
}
|
|
if !e.positions[0].exposure.IsZero() {
|
|
t.Errorf("received '%v' expected '%v'", e.positions[0].exposure, 0)
|
|
}
|
|
|
|
e = nil
|
|
err = e.Liquidate(decimal.Zero, tt)
|
|
if !errors.Is(err, common.ErrNilPointer) {
|
|
t.Error(err)
|
|
}
|
|
}
|
|
|
|
func TestPositionLiquidate(t *testing.T) {
|
|
t.Parallel()
|
|
item := asset.Futures
|
|
pair, err := currency.NewPairFromStrings("BTC", "1231")
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
p := &PositionTracker{
|
|
contractPair: pair,
|
|
asset: item,
|
|
exchange: testExchange,
|
|
PNLCalculation: &PNLCalculator{},
|
|
status: Open,
|
|
openingDirection: Long,
|
|
}
|
|
|
|
tt := time.Now()
|
|
err = p.TrackNewOrder(&Detail{
|
|
Date: tt,
|
|
Exchange: testExchange,
|
|
Pair: pair,
|
|
AssetType: item,
|
|
Side: Long,
|
|
OrderID: "lol",
|
|
Price: 1,
|
|
Amount: 1,
|
|
}, false)
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
|
|
err = p.Liquidate(decimal.Zero, time.Time{})
|
|
if !errors.Is(err, errCannotLiquidate) {
|
|
t.Error(err)
|
|
}
|
|
|
|
err = p.Liquidate(decimal.Zero, tt)
|
|
if !errors.Is(err, nil) {
|
|
t.Error(err)
|
|
}
|
|
|
|
if p.status != Liquidated {
|
|
t.Errorf("received '%v' expected '%v'", p.status, Liquidated)
|
|
}
|
|
if !p.exposure.IsZero() {
|
|
t.Errorf("received '%v' expected '%v'", p.exposure, 0)
|
|
}
|
|
|
|
p = nil
|
|
err = p.Liquidate(decimal.Zero, tt)
|
|
if !errors.Is(err, common.ErrNilPointer) {
|
|
t.Error(err)
|
|
}
|
|
}
|