mirror of
https://github.com/c9s/bbgo.git
synced 2024-11-10 09:11:55 +00:00
bfd9c8ac64
FEATURE: support recover FEATURE: add order into orderStore and recover position recover position/budget FEATURE: support recover budget
237 lines
9.9 KiB
Go
237 lines
9.9 KiB
Go
package dca2
|
|
|
|
import (
|
|
"context"
|
|
"math/rand"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/c9s/bbgo/pkg/bbgo"
|
|
"github.com/c9s/bbgo/pkg/core"
|
|
"github.com/c9s/bbgo/pkg/types"
|
|
"github.com/stretchr/testify/assert"
|
|
)
|
|
|
|
func generateOrder(side types.SideType, status types.OrderStatus, createdAt time.Time) types.Order {
|
|
return types.Order{
|
|
OrderID: rand.Uint64(),
|
|
SubmitOrder: types.SubmitOrder{
|
|
Side: side,
|
|
},
|
|
Status: status,
|
|
CreationTime: types.Time(createdAt),
|
|
}
|
|
|
|
}
|
|
|
|
func Test_GetCurrenctAndLastRoundOrders(t *testing.T) {
|
|
assert := assert.New(t)
|
|
|
|
t.Run("case 1", func(t *testing.T) {
|
|
now := time.Now()
|
|
openOrders := []types.Order{
|
|
generateOrder(types.SideTypeSell, types.OrderStatusNew, now),
|
|
}
|
|
|
|
closedOrders := []types.Order{
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-1*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-2*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-3*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-4*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-5*time.Second)),
|
|
}
|
|
|
|
currentRound, err := getCurrentRoundOrders(false, openOrders, closedOrders, 0)
|
|
|
|
assert.NoError(err)
|
|
assert.NotEqual(0, currentRound.TakeProfitOrder.OrderID)
|
|
assert.Equal(5, len(currentRound.OpenPositionOrders))
|
|
})
|
|
|
|
t.Run("case 2", func(t *testing.T) {
|
|
now := time.Now()
|
|
openOrders := []types.Order{
|
|
generateOrder(types.SideTypeSell, types.OrderStatusNew, now),
|
|
}
|
|
|
|
closedOrders := []types.Order{
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-1*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-2*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-3*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-4*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-5*time.Second)),
|
|
generateOrder(types.SideTypeSell, types.OrderStatusFilled, now.Add(-6*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-7*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-8*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-9*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-10*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-11*time.Second)),
|
|
generateOrder(types.SideTypeSell, types.OrderStatusFilled, now.Add(-12*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-13*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-14*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-15*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-16*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-17*time.Second)),
|
|
}
|
|
|
|
currentRound, err := getCurrentRoundOrders(false, openOrders, closedOrders, 0)
|
|
|
|
assert.NoError(err)
|
|
assert.NotEqual(0, currentRound.TakeProfitOrder.OrderID)
|
|
assert.Equal(5, len(currentRound.OpenPositionOrders))
|
|
})
|
|
}
|
|
|
|
type MockQueryOrders struct {
|
|
OpenOrders []types.Order
|
|
ClosedOrders []types.Order
|
|
}
|
|
|
|
func (m *MockQueryOrders) QueryOpenOrders(ctx context.Context, symbol string) ([]types.Order, error) {
|
|
return m.OpenOrders, nil
|
|
}
|
|
|
|
func (m *MockQueryOrders) QueryClosedOrdersDesc(ctx context.Context, symbol string, since, until time.Time, lastOrderID uint64) ([]types.Order, error) {
|
|
return m.ClosedOrders, nil
|
|
}
|
|
|
|
func Test_RecoverState(t *testing.T) {
|
|
assert := assert.New(t)
|
|
symbol := "BTCUSDT"
|
|
|
|
t.Run("new strategy", func(t *testing.T) {
|
|
openOrders := []types.Order{}
|
|
currentRound := Round{}
|
|
activeOrderBook := bbgo.NewActiveOrderBook(symbol)
|
|
orderStore := core.NewOrderStore(symbol)
|
|
state, err := recoverState(context.Background(), symbol, false, 5, openOrders, currentRound, activeOrderBook, orderStore, 0)
|
|
assert.NoError(err)
|
|
assert.Equal(WaitToOpenPosition, state)
|
|
})
|
|
|
|
t.Run("at open position stage and no filled order", func(t *testing.T) {
|
|
now := time.Now()
|
|
openOrders := []types.Order{
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusPartiallyFilled, now.Add(-1*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusNew, now.Add(-2*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusNew, now.Add(-3*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusNew, now.Add(-4*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusNew, now.Add(-5*time.Second)),
|
|
}
|
|
currentRound := Round{
|
|
OpenPositionOrders: openOrders,
|
|
}
|
|
orderStore := core.NewOrderStore(symbol)
|
|
activeOrderBook := bbgo.NewActiveOrderBook(symbol)
|
|
state, err := recoverState(context.Background(), symbol, false, 5, openOrders, currentRound, activeOrderBook, orderStore, 0)
|
|
assert.NoError(err)
|
|
assert.Equal(OpenPositionReady, state)
|
|
})
|
|
|
|
t.Run("at open position stage and there at least one filled order", func(t *testing.T) {
|
|
now := time.Now()
|
|
openOrders := []types.Order{
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusNew, now.Add(-2*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusNew, now.Add(-3*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusNew, now.Add(-4*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusNew, now.Add(-5*time.Second)),
|
|
}
|
|
currentRound := Round{
|
|
OpenPositionOrders: []types.Order{
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-1*time.Second)),
|
|
openOrders[0],
|
|
openOrders[1],
|
|
openOrders[2],
|
|
openOrders[3],
|
|
},
|
|
}
|
|
orderStore := core.NewOrderStore(symbol)
|
|
activeOrderBook := bbgo.NewActiveOrderBook(symbol)
|
|
state, err := recoverState(context.Background(), symbol, false, 5, openOrders, currentRound, activeOrderBook, orderStore, 0)
|
|
assert.NoError(err)
|
|
assert.Equal(OpenPositionOrderFilled, state)
|
|
})
|
|
|
|
t.Run("open position stage finish, but stop at cancelling", func(t *testing.T) {
|
|
now := time.Now()
|
|
openOrders := []types.Order{
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusNew, now.Add(-5*time.Second)),
|
|
}
|
|
currentRound := Round{
|
|
OpenPositionOrders: []types.Order{
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-1*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusCanceled, now.Add(-2*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusCanceled, now.Add(-3*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusCanceled, now.Add(-4*time.Second)),
|
|
openOrders[0],
|
|
},
|
|
}
|
|
orderStore := core.NewOrderStore(symbol)
|
|
activeOrderBook := bbgo.NewActiveOrderBook(symbol)
|
|
state, err := recoverState(context.Background(), symbol, false, 5, openOrders, currentRound, activeOrderBook, orderStore, 0)
|
|
assert.NoError(err)
|
|
assert.Equal(OpenPositionOrdersCancelling, state)
|
|
})
|
|
|
|
t.Run("open-position orders are cancelled", func(t *testing.T) {
|
|
now := time.Now()
|
|
openOrders := []types.Order{}
|
|
currentRound := Round{
|
|
OpenPositionOrders: []types.Order{
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-1*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusCanceled, now.Add(-2*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusCanceled, now.Add(-3*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusCanceled, now.Add(-4*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusCanceled, now.Add(-5*time.Second)),
|
|
},
|
|
}
|
|
orderStore := core.NewOrderStore(symbol)
|
|
activeOrderBook := bbgo.NewActiveOrderBook(symbol)
|
|
state, err := recoverState(context.Background(), symbol, false, 5, openOrders, currentRound, activeOrderBook, orderStore, 0)
|
|
assert.NoError(err)
|
|
assert.Equal(OpenPositionOrdersCancelled, state)
|
|
})
|
|
|
|
t.Run("at take profit stage, and not filled yet", func(t *testing.T) {
|
|
now := time.Now()
|
|
openOrders := []types.Order{
|
|
generateOrder(types.SideTypeSell, types.OrderStatusNew, now),
|
|
}
|
|
currentRound := Round{
|
|
TakeProfitOrder: openOrders[0],
|
|
OpenPositionOrders: []types.Order{
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-1*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusCanceled, now.Add(-2*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusCanceled, now.Add(-3*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusCanceled, now.Add(-4*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusCanceled, now.Add(-5*time.Second)),
|
|
},
|
|
}
|
|
orderStore := core.NewOrderStore(symbol)
|
|
activeOrderBook := bbgo.NewActiveOrderBook(symbol)
|
|
state, err := recoverState(context.Background(), symbol, false, 5, openOrders, currentRound, activeOrderBook, orderStore, 0)
|
|
assert.NoError(err)
|
|
assert.Equal(TakeProfitReady, state)
|
|
})
|
|
|
|
t.Run("at take profit stage, take-profit order filled", func(t *testing.T) {
|
|
now := time.Now()
|
|
openOrders := []types.Order{}
|
|
currentRound := Round{
|
|
TakeProfitOrder: generateOrder(types.SideTypeSell, types.OrderStatusFilled, now),
|
|
OpenPositionOrders: []types.Order{
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusFilled, now.Add(-1*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusCanceled, now.Add(-2*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusCanceled, now.Add(-3*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusCanceled, now.Add(-4*time.Second)),
|
|
generateOrder(types.SideTypeBuy, types.OrderStatusCanceled, now.Add(-5*time.Second)),
|
|
},
|
|
}
|
|
orderStore := core.NewOrderStore(symbol)
|
|
activeOrderBook := bbgo.NewActiveOrderBook(symbol)
|
|
state, err := recoverState(context.Background(), symbol, false, 5, openOrders, currentRound, activeOrderBook, orderStore, 0)
|
|
assert.NoError(err)
|
|
assert.Equal(WaitToOpenPosition, state)
|
|
})
|
|
}
|