2023-08-28 14:51:31 +08:00

58 lines
1.4 KiB
Go

package limit
import (
"context"
"errors"
"time"
"golang.org/x/time/rate"
)
const (
ConsumeBusyError = "consume failed because your request is too busy"
ConsumeTimeoutError = "consume failed because request deadline or waited timeout"
)
type LimitServiceProvider struct {
limiter *rate.Limiter
}
func NewLimitService(option LimitOption) LimitService {
return &LimitServiceProvider{
limiter: rate.NewLimiter(
rate.Every(time.Duration(option.LimitMillisecond)*time.Millisecond),
option.Burst,
),
}
}
func (provider *LimitServiceProvider) GetOption() LimitOption {
return LimitOption{
Burst: provider.limiter.Burst(),
LimitMillisecond: int64(1000 / provider.limiter.Limit()),
}
}
func (provider *LimitServiceProvider) SetOption(option LimitOption) {
provider.limiter.SetLimit(rate.Every(time.Duration(option.LimitMillisecond) * time.Millisecond))
provider.limiter.SetBurst(option.Burst)
}
func (provider *LimitServiceProvider) Consume() error {
if provider.limiter.Allow() {
return nil
}
return errors.New(ConsumeBusyError)
}
func (provider *LimitServiceProvider) ConsumeWithContext(ctx context.Context) error {
if err := provider.limiter.Wait(ctx); err != nil {
return errors.New(ConsumeTimeoutError)
}
return nil
}
func (provider *LimitServiceProvider) Clone() LimitService {
return NewLimitService(provider.GetOption())
}