mirror of
https://github.com/XTLS/Xray-core.git
synced 2024-11-30 00:56:03 +03:00
63 lines
1.4 KiB
Go
63 lines
1.4 KiB
Go
|
package drain
|
||
|
|
||
|
import (
|
||
|
"io"
|
||
|
"io/ioutil"
|
||
|
|
||
|
"github.com/xtls/xray-core/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 {
|
||
|
_, err := io.CopyN(ioutil.Discard, reader, int64(n))
|
||
|
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{}
|
||
|
}
|