Skip to content

Commit

Permalink
Merge pull request #11517 from vegaprotocol/11516
Browse files Browse the repository at this point in the history
fix: order spam check for amends fixed
  • Loading branch information
jeremyletang authored Jul 31, 2024
2 parents a99d0bb + 7ab7418 commit d220ac2
Show file tree
Hide file tree
Showing 3 changed files with 63 additions and 83 deletions.
2 changes: 1 addition & 1 deletion CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@
### 🐛 Fixes

- [11513](https://github.com/vegaprotocol/vega/issues/11513) - Rollback CometBFT to version `v0.38.8`.

- [11516](https://github.com/vegaprotocol/vega/issues/11516) - Fix order spam check for amends.

## 0.77.4

Expand Down
77 changes: 32 additions & 45 deletions core/execution/future/market.go
Original file line number Diff line number Diff line change
Expand Up @@ -5281,81 +5281,68 @@ func (m *Market) emitPartyMarginModeUpdated(ctx context.Context, party string, m
m.broker.Send(events.NewPartyMarginModeUpdatedEvent(ctx, e))
}

func (m *Market) checkOrderAmendForSpam(order *types.Order) error {
rf := num.DecimalOne()

factor := m.mkt.LinearSlippageFactor
if m.risk.IsRiskFactorInitialised() {
if order.Side == types.SideBuy {
rf = m.risk.GetRiskFactors().Long
} else {
rf = m.risk.GetRiskFactors().Short
}
}
var price *num.Uint
if order.PeggedOrder == nil {
price, _ = num.UintFromDecimal(order.Price.ToDecimal().Mul(m.priceFactor))
} else {
priceInMarket, _ := num.UintFromDecimal(m.getCurrentMarkPrice().ToDecimal().Div(m.priceFactor))
if order.Side == types.SideBuy {
priceInMarket.AddSum(order.PeggedOrder.Offset)
} else {
priceInMarket = priceInMarket.Sub(priceInMarket, order.PeggedOrder.Offset)
}
price, _ = num.UintFromDecimal(priceInMarket.ToDecimal().Mul(m.priceFactor))
}
margins := num.UintZero().Mul(price, num.NewUint(order.TrueRemaining())).ToDecimal().Div(m.positionFactor)
assetQuantum, err := m.collateral.GetAssetQuantum(m.settlementAsset)
if err != nil {
return err
}
if margins.Mul(rf.Add(factor)).Div(assetQuantum).LessThan(m.minMaintenanceMarginQuantumMultiplier.Mul(assetQuantum)) {
return fmt.Errorf("order value is less than minimum maintenance margin for spam")
}
return nil
}

func (m *Market) CheckOrderSubmissionForSpam(orderSubmission *types.OrderSubmission, party string, quantumMultiplier num.Decimal) error {
func (m *Market) checkOrderForSpam(side types.Side, orderPrice *num.Uint, orderSize uint64, peggedOrder *types.PeggedOrder, orderType vegapb.Order_Type, quantumMultiplier num.Decimal) error {
rf := num.DecimalOne()

factor := m.mkt.LinearSlippageFactor
if m.risk.IsRiskFactorInitialised() {
if orderSubmission.Side == types.SideBuy {
if side == types.SideBuy {
rf = m.risk.GetRiskFactors().Long
} else {
rf = m.risk.GetRiskFactors().Short
}
}

var price *num.Uint
if orderSubmission.PeggedOrder != nil || orderSubmission.Type == vega.Order_TYPE_MARKET {
if peggedOrder != nil || orderType == vega.Order_TYPE_MARKET {
priceInMarket, _ := num.UintFromDecimal(m.getCurrentMarkPrice().ToDecimal().Div(m.priceFactor))
offset := num.UintZero()
if orderSubmission.PeggedOrder != nil {
offset = orderSubmission.PeggedOrder.Offset
if peggedOrder != nil {
offset = peggedOrder.Offset
}
if orderSubmission.Side == types.SideBuy {
if side == types.SideBuy {
priceInMarket.AddSum(offset)
} else {
priceInMarket = priceInMarket.Sub(priceInMarket, offset)
}
price, _ = num.UintFromDecimal(priceInMarket.ToDecimal().Mul(m.priceFactor))
} else {
price, _ = num.UintFromDecimal(orderSubmission.Price.ToDecimal().Mul(m.priceFactor))
price, _ = num.UintFromDecimal(orderPrice.ToDecimal().Mul(m.priceFactor))
}

margins := num.UintZero().Mul(price, num.NewUint(orderSubmission.Size)).ToDecimal().Div(m.positionFactor)
margins := num.UintZero().Mul(price, num.NewUint(orderSize)).ToDecimal().Div(m.positionFactor)

assetQuantum, err := m.collateral.GetAssetQuantum(m.settlementAsset)
if err != nil {
return err
}
if margins.Mul(rf.Add(factor)).LessThan(quantumMultiplier.Mul(assetQuantum)) {
return fmt.Errorf("order value is less than minimum maintenance margin for spam")
value := margins.Mul(rf.Add(factor))
required := quantumMultiplier.Mul(assetQuantum)
if value.LessThan(required) {
return fmt.Errorf(fmt.Sprintf("order value (%s) is less than minimum maintenance margin for spam (%s)", value.String(), required.String()))
}
return nil
}

func (m *Market) checkOrderAmendForSpam(order *types.Order) error {
return m.checkOrderForSpam(
order.Side,
order.Price,
order.Size,
order.PeggedOrder,
order.Type,
m.minMaintenanceMarginQuantumMultiplier)
}

func (m *Market) CheckOrderSubmissionForSpam(orderSubmission *types.OrderSubmission, party string, quantumMultiplier num.Decimal) error {
return m.checkOrderForSpam(
orderSubmission.Side,
orderSubmission.Price,
orderSubmission.Size,
orderSubmission.PeggedOrder,
orderSubmission.Type,
quantumMultiplier)
}

func (m *Market) GetFillPrice(volume uint64, side types.Side) (*num.Uint, error) {
return m.matching.GetFillPrice(volume, side)
}
Expand Down
67 changes: 30 additions & 37 deletions core/execution/spot/market.go
Original file line number Diff line number Diff line change
Expand Up @@ -3323,62 +3323,55 @@ type IDGen interface {
NextID() string
}

func (m *Market) checkOrderAmendForSpam(order *types.Order) error {
assetQuantum, err := m.collateral.GetAssetQuantum(m.quoteAsset)
if err != nil {
return err
}

var price *num.Uint
if order.PeggedOrder == nil {
price, _ = num.UintFromDecimal(order.Price.ToDecimal().Mul(m.priceFactor))
} else {
priceInMarket, _ := num.UintFromDecimal(m.getCurrentMarkPrice().ToDecimal().Div(m.priceFactor))
if order.Side == types.SideBuy {
priceInMarket.AddSum(order.PeggedOrder.Offset)
} else {
priceInMarket = priceInMarket.Sub(priceInMarket, order.PeggedOrder.Offset)
}
price, _ = num.UintFromDecimal(priceInMarket.ToDecimal().Mul(m.priceFactor))
}

minQuantum := assetQuantum.Mul(m.minHoldingQuantumMultiplier)
value := num.UintZero().Mul(num.NewUint(order.Size), price).ToDecimal()
value = value.Div(m.positionFactor).Div(assetQuantum)
if value.LessThan(minQuantum.Mul(assetQuantum)) {
return fmt.Errorf("order value is less than minimum holding requirement for spam")
}
return nil
}

func (m *Market) CheckOrderSubmissionForSpam(orderSubmission *types.OrderSubmission, party string, quantumMultiplier num.Decimal) error {
func (m *Market) checkOrderForSpam(side types.Side, orderPrice *num.Uint, orderSize uint64, peggedOrder *types.PeggedOrder, orderType vega.Order_Type, quantumMultiplier num.Decimal) error {
assetQuantum, err := m.collateral.GetAssetQuantum(m.quoteAsset)
if err != nil {
return err
}

var price *num.Uint
if orderSubmission.PeggedOrder != nil || orderSubmission.Type == vega.Order_TYPE_MARKET {
if peggedOrder != nil || orderType == vega.Order_TYPE_MARKET {
priceInMarket, _ := num.UintFromDecimal(m.getCurrentMarkPrice().ToDecimal().Div(m.priceFactor))
offset := num.UintZero()
if orderSubmission.PeggedOrder != nil {
offset = orderSubmission.PeggedOrder.Offset
if peggedOrder != nil {
offset = peggedOrder.Offset
}
if orderSubmission.Side == types.SideBuy {
if side == types.SideBuy {
priceInMarket.AddSum(offset)
} else {
priceInMarket = priceInMarket.Sub(priceInMarket, offset)
}
price, _ = num.UintFromDecimal(priceInMarket.ToDecimal().Mul(m.priceFactor))
} else {
price, _ = num.UintFromDecimal(orderSubmission.Price.ToDecimal().Mul(m.priceFactor))
price, _ = num.UintFromDecimal(orderPrice.ToDecimal().Mul(m.priceFactor))
}

minQuantum := assetQuantum.Mul(quantumMultiplier)
value := num.UintZero().Mul(num.NewUint(orderSubmission.Size), price).ToDecimal()
value := num.UintZero().Mul(num.NewUint(orderSize), price).ToDecimal()
value = value.Div(m.positionFactor)
if value.LessThan(minQuantum.Mul(assetQuantum)) {
return fmt.Errorf("order value is less than minimum holding requirement for spam")
required := minQuantum.Mul(assetQuantum)
if value.LessThan(required) {
return fmt.Errorf(fmt.Sprintf("order value (%s) is less than minimum holding requirement for spam (%s)", value.String(), required.String()))
}
return nil
}

func (m *Market) checkOrderAmendForSpam(order *types.Order) error {
return m.checkOrderForSpam(
order.Side,
order.Price,
order.Size,
order.PeggedOrder,
order.Type,
m.minHoldingQuantumMultiplier)
}

func (m *Market) CheckOrderSubmissionForSpam(orderSubmission *types.OrderSubmission, party string, quantumMultiplier num.Decimal) error {
return m.checkOrderForSpam(
orderSubmission.Side,
orderSubmission.Price,
orderSubmission.Size,
orderSubmission.PeggedOrder,
orderSubmission.Type,
quantumMultiplier)
}

0 comments on commit d220ac2

Please sign in to comment.