2021-04-30 07:08:15 -04:00
|
|
|
package drain
|
|
|
|
|
|
|
|
import (
|
|
|
|
"io"
|
|
|
|
|
|
|
|
"github.com/v2fly/v2ray-core/v4/common/dice"
|
|
|
|
)
|
|
|
|
|
|
|
|
type BehaviorSeedLimitedDrainer struct {
|
|
|
|
DrainSize int
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewBehaviorSeedLimitedDrainer(behaviorSeed int64, drainFoundation, maxBaseDrainSize, maxRandDrain int) (Drainer, error) {
|
|
|
|
behaviorRand := dice.NewDeterministicDice(behaviorSeed)
|
|
|
|
BaseDrainSize := behaviorRand.Roll(maxBaseDrainSize)
|
|
|
|
RandDrainMax := behaviorRand.Roll(maxRandDrain) + 1
|
|
|
|
RandDrainRolled := dice.Roll(RandDrainMax)
|
|
|
|
DrainSize := drainFoundation + BaseDrainSize + RandDrainRolled
|
|
|
|
return &BehaviorSeedLimitedDrainer{DrainSize: DrainSize}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *BehaviorSeedLimitedDrainer) AcknowledgeReceive(size int) {
|
|
|
|
d.DrainSize -= size
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *BehaviorSeedLimitedDrainer) Drain(reader io.Reader) error {
|
|
|
|
if d.DrainSize > 0 {
|
|
|
|
err := drainReadN(reader, d.DrainSize)
|
|
|
|
if err == nil {
|
|
|
|
return newError("drained connection")
|
|
|
|
}
|
|
|
|
return newError("unable to drain connection").Base(err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func drainReadN(reader io.Reader, n int) error {
|
2021-09-27 18:02:19 -04:00
|
|
|
_, err := io.CopyN(io.Discard, reader, int64(n))
|
2021-04-30 07:08:15 -04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func WithError(drainer Drainer, reader io.Reader, err error) error {
|
|
|
|
drainErr := drainer.Drain(reader)
|
|
|
|
if drainErr == nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return newError(drainErr).Base(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
type NopDrainer struct{}
|
|
|
|
|
|
|
|
func (n NopDrainer) AcknowledgeReceive(size int) {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n NopDrainer) Drain(reader io.Reader) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewNopDrainer() Drainer {
|
|
|
|
return &NopDrainer{}
|
|
|
|
}
|